A CFRunLoop object

本文深入解析了CFRunLoop机制,介绍了如何使用CFRunLoop监控任务输入源并进行调度控制。此外,文章还探讨了sources、timers和observers三类对象的作用,并解释了不同运行模式的概念及其在实际应用中的意义。

 

 CFRunLoop 对象监控任务(task)的输入源,并在它们为处理做好准备的时候调度控制。输入源样例可能包括用户输入设备、网络链接、定期或时间延迟事件,还有异步回调。 

有3类对象可以被run loop监控:sources、timers、observers。

当这些对象需要处理的时候,为了接收回调,首先必须通过 CFRunLoopAddSource,CFRunLoopAddTimer, or CFRunLoopAddObserver把这些对象放入run loop。 要停止接收它的回调,你也可以稍候通过CFRunLoopRemoveSource从run loop中移除某个对象。 

run loop有不同的运行模式,每种模式都有其自身的对象集,run loop监控,同时在该模式下运行。 Core Foundation定义了一种默认模式kCFRunLoopDefaultMode来持有对象,在应用或线程闲置的时候这些对象应该被监控。当一个对象被添加到不认识的模式时,额外模式自动创建。每个run loop有它自己独立的模式集。

 

 

Core Foundation还定义了一个特殊的伪模式kCFRunLoopCommonModes来持有应当被“common”模式集共享的对象。 通过调用CFRunLoopAddCommonMode来添加一个模式到“common”模式集。 默认模式kCFRunLoopDefaultMode 总是common模式集中的一个成员。kCFRunLoopCommonModes 常数决不会传给CFRunLoopRunInMode。每个run loop有它自己独立的common模式集。

 

每个线程恰好有一个run loop,既不可以创建,也不能销毁线程的run loop。,Core Foundation 根据需要为你创建。通过CFRunLoopGetMain 你可以获得当前线程的run loop。调用lCFRunLoopRun 来使当前线程的run loop以默认模式运行起 来,直到调用CFRunLoopStop来停止run loop。你也可以调用CFRunLoopRunInMode来使当前线程的run loop以指定模式运行起来一段时间或者直到run loop被停止。 run loop只能在请求模式至少有一个source或者timer可监控的情况下运行起来。 

 

run loop可以递归运行,你可以在任何run loop 标注内部调用CFRunLoopRun 或 CFRunLoopRunInMode,还可以创建嵌套run loop,并在当前线程调用栈激活,在标注内并没有限制在那种模式可以运行。你可以创建另一个run loop,激活运行在任何可行的run loop模式,包括任何已经运行在调用堆栈中的更高的模式。

 

Cocoa 和 Carbon 每个都是建立在 CFRunLoop上来实现它们自己更高级别的事件循环。当编写一个  Cocoa 或者 Carbon 应用,你可以添加你的sources、timer和observers到它们的run loop对象中。你的对象将会作为常规应用事件循环的一部分来得到监控。使用 NSRunLoop 实例方法 getCFRunLoop 来获得对应应于cocoa run loop的CFRunLoop,在carbon应用中使用 GetCFRunLoopFromEventLoop 函数.

 

Runloop的工作:如何管理事件/消息,如何让线程在没有处理消息时休眠以避免资源占用、在有消息到来时立刻被唤醒。 Osx/ios提供了两种runloop对象 : CFRunLoopRef是在CoreFoundation框架呢,是纯C函数的API,是线程安全的 NSRunLoop是基于CFRunLoopRef的封装,不是线程安全 CFRunLoopRef的线程安全性:CFRunLoopRef不是在线程刚创建的时候创建的,而是在线程获取的时候创建的,如果不主动获取线程,一直不会有。CFRunLoopRef在创建时通过static CFSpinLock_t loopsLock锁住CFRunLoopRef对象,所以CFRunLoopRef是安全的。 一个runloop包含若干个mode。每个mode又包含若干个Source/Timer/Observer CFRunLoopSourceRef是事件产生的地方。Source有两个版本: Source0和Source1。 Source0只包含一个回调,它不能主动出发事件,需要调用CFRunLoopSOurceSignal(source)将source标记,然后手动调用CFRunloopWakeUp()来唤醒RunLoop Source1包含了一个mach_port和一个回调,被用于通过内核和其他线程来发送消息,这种source能主动唤醒runloop。 CFRunLoopTimerRef是基于时间的触发器,它和NSTimer是toll-free bridged的,可以混用,包含一个时间长度和一个回调。当其加入到RunLoop时,RunLoop会注册对应的时间点,当时间点到时,RunLoop会被唤醒执行回调。 CFRunLoopObserverRef是观察者,每个Observer都包含了一个回调,当RunLoop的状态发生变化时,观察者就能通过回调接受到这个变化。
用户升级ios26系统, 使用过程中LSSafeProtector这个第三方库的NSNotificationCenter报错, 如何解决, 中文方式说明, 下面是报错的方法以及错误信息 -(void)safe_addObserver:(id)observer selector:(SEL)aSelector name:(NSNotificationName)aName object:(id)anObject { [observer setIsNotification:YES]; [observer safe_changeDidDeallocSignal]; [self safe_addObserver:observer selector:aSelector name:aName object:anObject]; } #0 0x00000002478cc0cc in __pthread_kill () #1 0x00000001fa931810 in pthread_kill () #2 0x00000001ab154f1c in abort () #3 0x000000019cc43f60 in swift::fatalErrorv () #4 0x000000019cc43f80 in swift::fatalError () #5 0x000000019ccb19e4 in -[_TtCs12_SwiftObject doesNotRecognizeSelector:] () #6 0x000000019fb1f4f8 in ___forwarding___ () #7 0x000000019fb273a0 in _CF_forwarding_prep_0 () #8 0x000000010ea4db04 in -[NSNotificationCenter(Safe) safe_addObserver:selector:name:object:] at /Users/clouder/xiaohui-ios/powerone/3rdParty/LSSafeProtector/Foundation/NSNotificationCenter+Safe.m:103 #9 0x00000001a55f5e70 in ___lldb_unnamed_symbol297340 () #10 0x00000001a544dc4c in ___lldb_unnamed_symbol294151 () #11 0x00000001a544db1c in ___lldb_unnamed_symbol294149 () #12 0x00000001a54512b4 in ___lldb_unnamed_symbol294184 () #13 0x00000001a53eb2e4 in ___lldb_unnamed_symbol293679 () #14 0x00000001a544d39c in -[_UIAppCACommitFuture _invoke] () #15 0x00000001a05485bc in CA::Transaction::run_commit_handlers () #16 0x00000001a05090ac in CA::Context::commit_transaction () #17 0x00000001a0533ab0 in CA::Transaction::commit () #18 0x00000001a05413c0 in CA::Transaction::flush_as_runloop_observer () #19 0x00000001a543f0f0 in _UIApplicationFlushCATransaction () #20 0x00000001a543f024 in __setupUpdateSequence_block_invoke_2 () #21 0x00000001a544cee8 in _UIUpdateSequenceRunNext () #22 0x00000001a544c378 in schedulerStepScheduledMainSectionContinue () #23 0x0000000289db35f8 in UC::DriverCore::continueProcessing () #24 0x000000019fb43230 in __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__ () #25 0x000000019fb431a4 in __CFRunLoopDoSource0 () #26 0x000000019fb20c6c in __CFRunLoopDoSources0 () #27 0x000000019faf68b0 in __CFRunLoopRun () #28 0x000000019faf5c44 in _CFRunLoopRunSpecificWithOptions () #29 0x000000023eec6498 in GSEventRunModal () #30 0x00000001a5470ddc in -[UIApplication _run] () #31 0x00000001a5415b0c in UIApplicationMain ()
最新发布
09-26
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值