http://www.cocoachina.com/ios/20150601/11970.html
这种模型的关键点在于:如何管理事件/消息,如何让线程在没有处理消息时休眠以避免资源占用,在有消息到来时立刻被唤醒。
提供了两个对象:NSRunloop和CFRunLoopRef
CFRunLoopSourceRef 是事件产生的地方。Source有两个版本:Source0 和 Source1。
苹果用Runloop实现的功能有:
autoreleasePool
事件响应
手势识别
界面更新
定时器
PerformSelector
关于GCD
关于网络请求
Runloop:实现这种模型的关键点在于:如何管理事件/消息,如何让线程在没有处理消息时休眠以避免资源占用、
在有消息到来时立刻被唤醒。
线程和RunLoop是一一对应的。
Runloop对外的接口
在CoreFoundation里面关于Runloop有5个类:
CGRunLoopRef//Runloop
CFRunLoopModeRef//RunLoop里面的mode
CFRunLoopSourceRef//RunLoop的mode里的资源
CFRunLoopTimerRef//RunLoop的mode里的定时器
CFRunLoopObserverRef//RunLoop的mode里的观察者
下面是关于5个类的介绍:
CFRunLoopSourceRef:是事件产生的地方。Source有两个版本:Source0和source1.
source0只包含了一个回调(函数指针),并不能主动触发事件,使用的时候需要先调用CFRunLoopSourceSignal(source)
手动调用CFRunLoopWakeUp(runloop)来唤醒RunLoop。
source1包含了一个mach_port和一个回调,用于通过内核和其他线程相互发送消息。这种
Source能主动唤醒RunLoop的线程。
CFRunLoopTimerRef:基于时间的触发器。
他和NSTimer是toll-free bridged的,可以混用。其包含一个时间长度和一个回调(函数指针)当其加入到RunLoop时,RunLoop会注册对应的时间点,当时间点到时,RunLoop会被唤醒以执行那个回调。
CFRunLoopObserverRef是观察者,每个Observer都包含一个回调,当RunLoop的状态发生变化时,观察者就能通过回调接受到这个变化,可以观察的时间点有以下几个:
typedef CF_OPTIONS(CFOptionFlags, CFRunLoopActivity) {
kCFRunLoopEntry = (1UL << 0),
// 即将进入Loop
kCFRunLoopBeforeTimers = (1UL << 1),
// 即将处理 Timer
kCFRunLoopBeforeSources = (1UL << 2),
// 即将处理 Source
kCFRunLoopBeforeWaiting = (1UL << 5),
// 即将进入休眠
kCFRunLoopAfterWaiting = (1UL << 6),
// 刚从休眠中唤醒
kCFRunLoopExit = (1UL << 7),
// 即将退出Loop
};
四种适合使用RunLoop的方法:
1)使用端口或自定义输入源和其他线程通信。
2)子线程中使用定时器。
3)cocoa中使用任何performSelector到了线程中运行方法
4)使线程履行周期性任务。
5)在子线程中使用NSURLConnection异步请求,也需要使用RunLoop,要不然子线程就会退出了.
主线程的RunLoop是默认开启的,但是子线程的RunLoop默认是关闭的,如果要在子线程中使用RunLoop需要自己手动开启.
CFRunLoopSource 是RunLoop的数据源抽象类 RunLoop定义了两个Version的Source:
1.Source0:处理App内部事件、App自己负责管理(触发),如UIEvent,CFSocket
2.Source1:有Runloop和内核管理,Mach port驱动,如CFMachPort、CFMessagePort
CFRunLoopObserver用于向外部报告RunLoop当前状态的更改,框架中很多机制都由RunLoopObserver触发,如CAAnimation
总结:如果使用while来等待分线程的完成,那么就会阻塞UI线程,如果用RunLoop来等待分线程的完成就不会阻塞UI线程。所以当主线程需要等待分线程完成的时候,用NSRunLoop,不能用while,会造成UI阻塞.
退出RunLoop的方法:
1.设置超时限定
2.通知RunLoop停止
线程安全:
1.Core Foundation中的函数石线程安全的
2.Cocoa的NSRunloop不是线程安全的,只在RunLoop所在线程改变Run Loop。如果添加源或者定时器到属于另一个线程的runloop,
程序会发生崩溃或发生意想不到的错误。
创建自定义输入源遵循下列步骤:
1.输入源要处理的信息
2.使感情趣的客户知道如何和输入源交互的调度程序
3.处理客户发送请求的程序
4.使输入源失效的取消程序
RunLoopSource对象管理着命令缓冲区并以此来接收其他线程的消息。
RunLoopContext对象是一个用于传递RunLoopSource对象和run loop引用给程序主线程的一个容器。
输入源的主要工作就是将与源关联的线程休眠,直到有事件发生。
配置基于端口的输入源:
配置NSMachPort对象