什么是 Runloop?
- 从字面意思上就是运行循环
- 它内部对应就是 do-while 循环,在这个循环内部不断地处理各种任务。
- 一个线程对应一个 runloop,主线程的 runloop 是默认启动的,子线程的 runloop 得手动启动(条用 run方法)
- runloop 只能选择一个 Mode 启动,如果当前 mode 没有任何 source、timer,那么就直接退出 runloop
- 基本作用就是保持程序的持续运行,处理 App 中的各种事件。
RunLoop 对象
- Foundation:NSRunLoop
- Core Foundation:CFRunLoopRef
RunLoop 与线程
- 每条线程都有唯一一个与之对应的 RunLoop 对象
- 主线程的 RunLoop 对象已经创建好了,子线程的 RunLoop需要主动创建
- RunLoop 在第一次获取时创建,在线程结束时销毁
获得 RunLoop 对象
- Foundation
- [NSRunLoop currentRunLoop]; //获取当前线程的 RunLoop
- [NSRunLoop mainRunLoop]; // 获取主线程的 RunLoop
- Core Foundation
- CFRunLoopGetCurrent();
- CFRunLoopGetMain();
CFRunLoopModeRef
- CFRunLoopModeRef代表 RunLoop 的运行模式
- 一个 RunLoop 包含若干个 mode,每个 mode 又包含若干个 source、timer、observer
- 每次启动 RunLoop 时,只能指定其中一个 mode,这个 mode 被称为currentMode
- 如果需要切换 mode,只能退出 RunLoop,在重新指定 mode
- 这样做主要是为了分隔开不同组的 source、timer、observer,让其不受影响
- mode 主要是用来指定事件在运行循环中的优先级,分为:
- NSDefaultRunLoopMode(kCFRunLoopDefaultMode):默认,空闲状态
- UITrackingRunLoopMode: ScrollView滚动的时候切换该 mode
- UITrackingRunLoopMode: RunLoop 启动时,会切换该 mode
- NSRunLoopCommonModes(kCFRunLoopCommonModes):任意的 mode
CFRunLoopTimerRef
- CFRunLoopTimerRef是基于时间的触发器
- CFRunLoopTimerRef基本上说就是 NSTimer,它受 RunLoop 的 mode 影响
- GCD 的定时器不受 RunLoop 的 mode 影响
CFRunLoopSourceRef
- CFRunLoopSourceRef 是事件源(输入源)
- port-based source
- custom input source
- coca perform selector source
CFRunLoopObserverRef
- CFRunLoopObserverRef 是观察者,能监听 RunLoop 的状态改变
- 可以监听的时间点有
- kCFRunLoopEntry 即将进入 loop
- kCFRunLoopBeforeTimers 即将处理 timer
- kCFRunLoopBeforeSources 即将处理 source
- kCFRunLoopBeforeWaiting 即将进入休眠
- kCFRunLoopAfterWaiting 从休眠中唤醒
- kCFRunLoopExit 即将退出
CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
NSLog(@"监听 runloop的状态改变 - %zd", activity);
});
CFRunLoopAddObserver(CFRunLoopGetCurrent(), observer, kCFRunLoopDefaultMode);
CFRelease(observer);
RunLoop 的应用
- NSTimer
- ImageView 的显示
- PerformSelector
- 常驻线程
- 自动释放池
RunLoop 定时源和输入源
- RunLoop 处理输入事件的两种不同源:输入源和定时源
- 输入源传递异步消息,通常来自于其他线程和程序
- 定时源则传递同步消息,在特定事件或者一定时间间隔发生
NSTimer 会失效。
- NSTimer 默认的模式是 NSDefaultRunLoopMode,默认是空闲的时候调用的,当出现 ScrollView 滚动的时候会出现,RunLoop 的 mode 会改变。
- 把 NSTimer 添加到NSRunLoopCommonModes就可以了
在开发中如何使用 RunLoop?什么应用场景?
- 开启一个常驻线程(让一个子线程不进入消亡状态,等他其他线程发来的消息,处理其他事件)
- 在子线程中开启一个定时器
- 在子线程中进行一些上期监控
- 可以控制定时器在特定模式下执行
- 可以让某事件(行为、任务)在特定下执行
- 可以添加 Observer 监听 RunLoop 的状态,比如监听点击事件的处理(在所有点击事件之前做一些事情)