- 运行循环
RunLoop的基本作用:
- 保持程序的持续运行
- 处理APP中各种事件(比如:触摸事件,定时器事件,Selector事件等)
- 能节省CPU资源,提高程序的性能:该做事的时候就唤醒,没有事情就睡眠
RunLoop对象
iOS中有2套API来访问和使用RunLoop
- Foundation框架中的NSRunLoop;
- Core Foundation中的CFRunLoop;
NSRunLoop是基于 CFRunLoopRef 的一层OC包装,所以要了解RunLoop内部结构,需要多研究CFRunLoopRef层面的API(Core Foundation层面)
RunLoop与线程
- 每条线程都有唯一的一个与之对应的RunLoop对象
- 主线程中的RunLoop由系统自动创建,子线程中RunLoop可以通过手动创建
- RunLoop在线程结束的时候会被销毁
获取RunLoop对象
-
Foundation框架中
[NSRunLoop currentRunLoop];//获得当前线程的RunLoop对象
[NSRunLoop mainRunLoop]; // 获得主线程的RunLoop对象 Core Foundation框架中
CFRunLoopGetCurrent(); // 获得当前线程的RunLoop对象
CFRunLoopGetMain(); // 获得主线程的RunLoop对象
RunLoop 内部结构
在 RunLoop 可以在不同模式下切换,在每个 Mode 下面都会有 Source Observer Timer,分别是 接收和处理消息源,例如触摸事件等,观察者,监控者执行的活动,例如 进入、time时间、休眠、唤醒等活动,time 主要是定时器相关的任务。
可以简单的通过下面的这张逻辑图了解。
我们以一个button点击时间为例,我们点击后,会依次执行,到 第五步,如果有 source 进入那么会跳到第九步处理唤醒时接收的消息,然后跳回第二步没有 timer 事件,就处理 source0 ,如果没有了事件,就会进行线程休眠。
如果有事件点击或是 time 事件,那么线程会被再次唤醒,执行上述的流程。
CFRunLoopSource
RunLoop的数据源抽象类(类似于OC中的protocol)
RunLoop定义了两个版本的source:Source0 和 Source1
Source0:处理的是App内部的事件、App自己负责管理,如按钮点击事件等。
Source1:由RunLoop和内核管理,Mach Port驱动,如CFMachPort、CFMessagePort
Source有两个版本:Source0 和 Source1。主要的差别是主动和手动的唤醒RunLoop
Source0:只包含了一个回调(函数指针),它并不能主动触发事件。使用时,你需要先调用CFRunLoopSourceSignal(source)
,将这个 Source 标记为待处理,然后手动调用CFRunLoopWakeUp(runloop)
来唤醒RunLoop
,让其处理这个事件。--->加入到RunLoop
得待处理队列中,手动唤醒RunLoop
Source1 :包含了一个 mach_port 和一个回调(函数指针),被用于通过内核和其他线程相互发送消息。这种 Source 能主动唤醒RunLoop
的线程