RunLoop

一、RunLoop的基本概念:

runloop从字面的意思来看就是:运行循环。

runloop的基本作用:

1、保持程序的持续运行

2、处理app中各种事件(触摸事件、定时器事件、Selector事件等)

3、能节省CPU,提高程序的性能:该做事的时候就被唤醒,没有事情就睡眠

假如没有了runloop,程序会在main函数执行完毕后退出,正是因为有了runloop,导致主函数没有马上退出,保证了程序持续运行。简单的可以理解为

二、RunLoop对象

iOS中有2套API来访问和使用RunLoop

1、Foundation框架中的NSRunLoop

2、Core Foundation中的CFRunLoop

NSRunLoop是基于CFRunLoop的一层OC包装。但是NSRunLoop不是线程安全的,而CFRunLoopRef是线程安全的。

三、RunLoop与线程

1、每条线程都有唯一的一个与之相对应的RunLoop对象

2、主线程的RunLoop由系统自动创建,子线程的RunLoop可以手动创建。

3、RunLoop在线程结束的时候会被销毁。

获取RunLoop对象

四、RunLoop的结构


首先我们需要知道的是CFRunLoop有五大类:

1、CFRunLoopRef 

2、CFRunLoopMod

3、CFRunLoopObserverRef

4、CFRunLoopSourceRef

5、CFRunLoopTimerRef

下边我们一一介绍上边的5大类:

1、CFRunLoopRef:一个RunLoop对象,没啥好解释的。

2、CFRunLoopMod。这个mode咱们好好聊聊。

CFRunLoopMod代表RunLoop的运行模式。

(1)一个RunLoop中包含多个Mode,每个Mode中又包含了多个Source/Timer/Observer。

(2)一个RunLoop在同一时间只能处在一种运行模式下,这个模式就是CurrentMode。

(3)如果切换Mode,只能退出当前的RunLoop,主要是为了分隔开不同组的Source/Timer/Observer。

系统默认注册了5种Mode

这里就遇到一个面试题:调用+scheduledTimerWithTimeInterval...的方式触发的timer,在滑动页面上的列表是,timer会暂停调用,为什么?如何解决的?

答:我们知道,一个timer在NSDefaultMode下被触发,如果这个时候拖动scrollview的话,这个timer就失效了,因为拖动scrollview,RunLoop的mode切换为UITrackingRunLoopMode。如果想要让一个定时器在两个模式下都有效有两种方法:1、将它加入到两个mode中;2、将timer加入到顶层的RunLoop的commonModeItems集合中,RunLoop会自动将这个集合中的所有item同步到具有"common"标示的mode。

第一种方案的代码:(是mainRunLoop还是currentMode还是需要自己做判断的)

第二种方案的代码:

3、CFRunLoopObserver:RunLoop状态的观察者,每一个观察者都包含一个回调(指针函数),当RunLoop的状态发生变化时,观察者就能通过回调接收这个变化。观察状态由以下几种:

有了这个观察者,就有了下边咱们要说到的RunLoop的处理逻辑。(这里是不是想到了ViewController的生命周期?)

4、CFRunLoopSourceRef:事件产生的地方

按照函数调用栈分为两类:Source0、Source1

Source0非基于Port的。只包含一个回调函数指针,使用时需要将事件标记为待处理:CFRunLoopSourceSignal(source),再调用CFRunLoopWakeUP(runloop)来唤醒RunLoop,使其处理整个事件

Source1基于Port的。通过内核和其他线程通信,接收、分发系统事件。

5、CFRunLoopTimerRef:基于时间的触发器

说到定时器,有一种说法,说RunLoop的timer和GCD中的timer是一个东西,其实不是的。CFRunLoopTimer基本上说的就是NSTimer,它受RunLoop Mode的影响。

GCD定时器不受RunLoop Mode的影响

五、RunLoop的处理逻辑

上图右边是线程的输入源:(这个输入源是不是和上边说的CFRunLoopSourceRef:事件产生的地方有什么联系啊)

(1)基于端口的输入源(Port Sources)

(2)自定义输入源(Custom Sources)

(3)Cocoa执行Selector的源(performSelectorxxx方法)

(4)定时源(Timer Sources)

线程针对上边不同的输入源,又不同的处理机制

(1)handlePort——处理基于端口的输入源

(2)customSrc——处理用户自定义输入源

(3)mySelector——处理Selector的源

(4)timerFired——处理定时源

上边是官方逻辑,下边的是非官方逻辑,从非官方逻辑里面我们可以看到我们上边说到的CFRunLoop的五大类都在什么时候用

六、RunLoop的具体使用

(1)事件传递与手势识别

对于硬件事件(触摸、锁屏、摇晃)的处理,苹果注册了一个基于port的source1,它的回调函数是__IOHIDEventSystemClientQueueCallback(),事件发生后,系统将事件包装成IOHIDEvent对象,并由mach port分配到对应的APP进程中,随后触发source1的回调,并调用_UIApplicationHandleEventQueueCallback()进行内部分发,其中包括识别 UIGesture/处理屏幕旋转/发送给 UIWindow 等,接下来发生的响应者链条了。

对于手势识别:当_UIApplicationHandleQueueCallback()接收到手势的时候,会将TouchBegin等事件的回调打断,随后会将这个手势标记为待处理状态,同时注册一个observer,检测BeforeWaiting状态,当RunLoop即将进入休眠时,其内部会获取到刚才所有标记为待处理的手势,执行_UIGestureRecognizerUpdateQueue()。

(2)Autorealease

iOS中autorelease变量什么时候释放,应该分为两种情况:

手动释放@autoreleasepool { }中的自动释放变量在当前大括号作用域结束时释放;

系统释放:在当前RunLoop本次Loop结束后释放;

autorelease原理:

(3)页面刷新

当在操作 UI 时,比如改变了 Frame、更新了 UIView/CALayer 的层次时,或者手动调用了 UIView/CALayer 的 setNeedsLayout/setNeedsDisplay方法后,这个 UIView/CALayer 就被标记为待处理,并被提交到一个全局的容器去。

苹果注册了一个 Observer 监听 BeforeWaiting(即将进入休眠) 和 Exit (即将退出Loop) 事件,回调去执行:

_ZN2CA11Transaction17observer_callbackEP19__CFRunLoopObservermPv()。这个函数里会遍历所有待处理的 UIView/CAlayer 以执行实际的绘制和调整,并更新 UI 界面。在这个函数之后就在屏幕上看到UI的变化。

图片刷新(假如界面要刷新N多图片需要渲染),此时用户拖拽UI控件就会出现卡的效果,我们可以通过RunLoop实现,只在RunLoop默认Mode下下载,也就是拖拽Mode下不刷新图片)

(4)Timer

可以说没有RunLoop就不可能实现定时器的功能。定时器的大致原理:设定一个时间点,将定时器加入RunLoop中,等到达设定的时间点的时候回唤醒线程处理回调。

(5)PerfromSeletor:afterDelay:

如果当前线程中没有RunLoop这个方法是不会有效的,本质上是在当前线程的RunLoop中添加一个定时器,当时间点到了会唤醒RunLoop执行回调。

(6)dispatch_main_queue

当调用dispatch_async(dispatch_get_main_queue(), block)时,libDispatch 会向主线程的 RunLoop 发送消息,RunLoop会被唤醒,并从消息中取得这个 block,并在回调__CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__()里执行这个 block。但这个逻辑仅限于 dispatch 到主线程,dispatch 到其他线程仍然是由 libDispatch 处理的。

(7)保证一个线程永远不死

结束。谢谢。





总的来说,这个RunLoop比RunTime要容易一些,可能是因为用到的地方比较多,另外能够看到,不像RunTime藏的很深的样子。

该问借鉴了:(1)RunLoop

                   (2)RunLoop

有兴趣的可以去看看这个:深入理解RunLoop

在这里感谢上边两位作者,如果有版权问题,可以联系我。谢谢。

最后,哪里不对的地方可以给我留言,我会及时改进的,谢谢大家。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 210,978评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 89,954评论 2 384
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,623评论 0 345
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,324评论 1 282
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,390评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,741评论 1 289
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,892评论 3 405
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,655评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,104评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,451评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,569评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,254评论 4 328
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,834评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,725评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,950评论 1 264
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,260评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,446评论 2 348

推荐阅读更多精彩内容