OC-RunLoop总结

什么是RunLoop?
RunLoop 的表象是保证线程一直存在的一个事件循环机制,本质是封装了一个可以接收消息处理事件并且无限dowhile循环的对象。

作用:
1、保持线程的存活
2、在主线程中,用于处理各种事件:

  1、设备上硬件的消息数据,比如:屏幕触摸,滑动,陀螺仪的反馈等
  2、手势识别,内核消息
  3、定时器(timer),"preformSelector"方法调用,block
  4、发送通知,通知订阅者目前的状态,订阅者根据不同的状态做出不同的反应,比如:界面刷新,autoreleasepool的释放

3、在保证线程存活的情况下,还能在没有任务的时候保持休眠状态,节省cpu资源,提高APP的运行性能。

RunLoop必须要运行在一个mode里,并且只能运行在一个mode里,如果有多个mode,想要切换mode,必须要先退出当前mode,然后才能进入到切换的mode中。

系统默认注册的5个Mode:
1. kCFRunLoopDefaultMode:App的默认Mode,通常主线程是在这个Mode下运行
2. UITrackingRunLoopMode:界面跟踪Mode,用于ScrollView追踪触摸滑动,保证界面滑动时不受其他 Mode 影响
3. UIInitializationRunLoopMode: 在刚启动 App 时第进入的第一个 Mode,启动完成后就不再使用,会切换到kCFRunLoopDefaultMode
4. GSEventReceiveRunLoopMode: 接受系统事件的内部 Mode,通常用不到
5. kCFRunLoopCommonModes: 这是一个占位用的Mode,作为标记kCFRunLoopDefaultMode和UITrackingRunLoopMode用,并不是一种真正的Mode 

Mode中包含的item
1、sources0、sources1
2、machport
3、timer
4、observers

mode中至少有一个存在RunLoop才不会退出。
而主线程是个例外,主线程RunLoop可以不需要这些条件就能运行,源码在判断的时候发现是主线程则直接return,跳过了条件检查阶段,也就是说主线程在任何情况下都不会退出

总结:RunLoop包含若干个Mode,每个Mode又包含若干个 Source/Timer/Observer,主线程RunLoop除外。

Runloop的创建过程

1、检查__CFRunLoops字典是否存在,不存在的话说明主线程runloop还未启动,则先创建主线程runloop,并且创建一个零时dic字典绑定主线程和主RunLoop(key是线程,value是RunLoop),然后将dic赋值给__CFRunLoops
2、检查__CFRunLoops字典中是否存在当前线程的RunLoop,没有则创建一个新的RunLoop,并且与当前线程绑定。
3、判断当前线程是否是创建runloop传入的线程,如果是则注册一个回调函数,该函数会在线程销毁的时候进行线程与RunLoop的解绑,其实就是将该线程的RunLoop从__CFRunLoops中删除

Mode的Sources Item分为2种,一种是sources0,一种是sources1
这2种sources的区别
1、sources0不能直接唤醒Runloop,只能标记该事件需要处理,要手动调用weakup来唤醒Runloop
2、sources1的结构体中有一个回调函数,是基于port端口监听的,可以直接唤醒Runloop来处理该事件

哪些情况下RunLoop会被唤醒

1、timer注册的时间到了
2、被sources1的端口消息唤醒
3、mode被停止的时候
4、添加timer的时候如果不是添加到当前Runloop则唤醒被添加的Runloop
5、RunLoop超时(实际上超时不会发生)以及停止的时候

RunLoop循环流程

1. 通知observer run loop被触发
2. 如果有timers事件的话,通知observer  
3. 如果有source0要处理的话,通知observer  
4. 触发所有的准备完毕的source0  
5. 如果当前是主线程的runloop,并且主线程有事儿,跳到第9步          
6. 通知Observer runloop将进入sleep状态  
7. mach进入sleep和监听状态  
8. 通知observer,runloop被woke up  
9. 如果runloop是被唤醒,CFRUNLOOP_WAKEUP_FOR_WAKEUP  
10. 如果用户定义的timer被触发,处理event并重启RunLoop  
11. 如果dispatchPort,处理主线程  
12. 如果一个source1被触发,__CFRunLoopDoSource1  
13. 继续循环或通知observer runloop将要exited。

你可能感兴趣的:(OC-RunLoop总结)