RunLoop的理解

今天同事问我RunLoop的知识。说他平时会用但是有些地方还是不清楚,所以我特意整理了一下,因为网上关于RunLoop的讲解很多,这里我写的是个人对RunLoop的理解,希望写的易懂一些让更多的人明白。

RunLoop基本作用

  • 保持程序的持续运行
  • 处理App中的各种事件(比如触摸事件、定时器事件、Selector事件)
  • 节省CPU资源,提高程序性能:该做事时做事,该休息时休息

程序一开始系统就为主线程创建了一个RunLoop,这也是一个app能一直运行的原因。

下面两句话很重要

  1. 每条线程都有唯一的一个与之对应的RunLoop对象
  2. 主线程的RunLoop已经自动创建好了,子线程的RunLoop需要主动创建

不管是主线程还是子线程都可以有个RunLoop 而且只能有一个。主线程的RunLoop在程序开始的时候已经创建好了,子线程的RunLoop需要我们自己去创建。

获取RunLoop的方式

//Foundation
[NSRunLoop currentRunLoop]; // 获得当前线程的RunLoop对象
[NSRunLoop mainRunLoop]; // 获得主线程的RunLoop对象
//Core Foundation
CFRunLoopGetCurrent(); // 获得当前线程的RunLoop对象
CFRunLoopGetMain(); // 获得主线程的RunLoop对象

苹果不允许直接创建 RunLoop,它只提供了两个自动获取的函数:NSRunLoop currentRunLoop( CFRunLoopGetMain( ) )和 NSRunLoop mainRunLoop( CFRunLoopGetCurrent( ) )。这两个函数内部的逻辑大概是下面这样:

/// 全局的Dictionary,key 是 pthread_t, value 是 CFRunLoopRef
static CFMutableDictionaryRef loopsDic;
/// 访问 loopsDic 时的锁
static CFSpinLock_t loopsLock;

/// 获取一个 pthread 对应的 RunLoop。
CFRunLoopRef _CFRunLoopGet(pthread_t thread) {
OSSpinLockLock(&loopsLock);

if (!loopsDic) {
    // 第一次进入时,初始化全局Dic,并先为主线程创建一个 RunLoop。
    loopsDic = CFDictionaryCreateMutable();
    CFRunLoopRef mainLoop = _CFRunLoopCreate();
    CFDictionarySetValue(loopsDic, pthread_main_thread_np(), mainLoop);
}

/// 直接从 Dictionary 里获取。
CFRunLoopRef loop = CFDictionaryGetValue(loopsDic, thread));

if (!loop) {
    /// 取不到时,创建一个
    loop = _CFRunLoopCreate();
    CFDictionarySetValue(loopsDic, thread, loop);
    /// 注册一个回调,当线程销毁时,顺便也销毁其对应的 RunLoop。
    _CFSetTSD(..., thread, loop, __CFFinalizeRunLoop);
}

OSSpinLockUnLock(&loopsLock);
return loop;
}

以上代码来源网络

子线程刚创建时并没有 RunLoop,如果你不主动获取,那它一直都不会有。RunLoop 的创建是发生在第一次获取时,RunLoop 的销毁是发生在线程结束时。

CFRunLoopModeRefbo(Mode)

RunLoop的Mode有五种:

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

ps:其实kCFRunLoopCommonModes标记了UITrackingRunLoopMode和kCFRunLoopDefaultMode两种模式,这里大家先了解一下后面会讲到。

这张图特别好,我就拿来用了

看上图,这张图描述的很清楚一个RunLoop可以包含多个Mode。

请务必记住下面的几句话?。。。?!

  • CFRunLoopModeRef代表RunLoop的运行模式

  • 一个 RunLoop 包含若干个 Mode,每个Mode又包含若干个Source/Timer/Observer

  • 每次RunLoop启动时,只能指定其中一个 Mode,这个Mode被称作 CurrentMode

  • 如果需要切换Mode,只能退出Loop,再重新指定一个Mode进入

  • 这样做主要是为了分隔开不同组的Source/Timer/Observer,让其互不影响

下面我们来举个例子

[NSTimer scheduledTimerWithTimeInterval:<#(NSTimeInterval)#> target:<#(nonnull id)#> selector:<#(nonnull SEL)#> userInfo:<#(nullable id)#> repeats:<#(BOOL)#>];

这个方法是创建一个NSTimer,并把它加入到RunLoopDefaultMode 模式下的RunLoop中。

NSTimer *time=[NSTimer timerWithTimeInterval:<#(NSTimeInterval)#> target:<#(nonnull id)#> selector:<#(nonnull SEL)#> userInfo:<#(nullable id)#> repeats:<#(BOOL)#>]

而这个方法只是创建了一个NSTimer

我们可以把这个timer加到一个RunLoop中

[[NSRunLoop currentRunLoop] addTimer:time forMode:NSDefaultRunLoopMode];

这样也可以实现上面代码实现的目的:创建一个NSTimer,并把它加入到RunLoopDefaultMode 模式下的RunLoop中。

现在有这样一个场景,添加一个定时器并把它加入到NSDefaultRunLoopMode,定时器正常运行,这时屏幕开始滑动,你会发现你的定时器不管用了,这个为什么呢?
这是因为在进行滑动的时候RunLoop进入了UITrackingRunLoopMode(忘记的请查看RunLoop的Mode)而你的time是在:NSDefaultRunLoopMode模式下的所以失效了当手一松开又进入了NSDefaultRunLoopMode,定时器又可以工作了现在该怎么办呢?
接下来我们按住NSDefaultRunLoopMode进入头文件看看发现

FOUNDATION_EXPORT NSString * const NSDefaultRunLoopMode;
FOUNDATION_EXPORT NSString * const NSRunLoopCommonModes NS_AVAILABLE(10_5, 2_0);

里面怎么就两个Mode,而且有一个还是用来占位的,别急我们打印一下在NSRunLoopCommonModes的RunLoop看看它的Mode是什么

 **common modes = <CFBasicHash 0x7f8009e035e0 [0x1095c5a40]>{type = mutable set, count = 2,**
 **entries =>**
 ** 0 : <CFString 0x10a4fc210 [0x1095c5a40]>{contents = "UITrackingRunLoopMode"}**
 ** 2 : <CFString 0x1095e65e0 [0x1095c5a40]>{contents = "kCFRunLoopDefaultMode"}**
 **}**

注意看**common modes **包含UITrackingRunLoopMode和kCFRunLoopDefaultMode
所以当你有一个定时器,能让它在屏幕滑动的情况下和正常情况下都还能使用就应该使用NSRunLoopCommonModes。
如果只让定时器在屏幕滑动时刻起作用该怎么办呢?
加入到UITrackingRunLoopMode模式下不就行了,哈哈。

学到这里大家应该知道一个RunLoop可以有多个Mode,但是在每次RunLoop启动时,只能指定其中一个 Mode

这张图特别好,我就拿来用了

如果感觉这篇文章对您有所帮助,顺手点个喜欢,谢谢啦

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

推荐阅读更多精彩内容

  • 什么情况下使用runloop? runloop好比就是跑圈,就是一个线程一直在做某一件事情。 一般主线程会自动运行...
    进击的小杰阅读 4,410评论 4 7
  • Run loop 剖析:Runloop 接收的输入事件来自两种不同的源:输入源(intput source)和定时...
    Mitchell阅读 12,426评论 17 111
  • 每个线程有一个消息循环 —> 消息循环监听着输入事件—> 事件有两种类型 输入源和定时源 —> 将创建好的输入源以...
    金歌漫舞阅读 364评论 0 0
  • RunLoop的概念 一般来说,一个线程一次只能执行一个任务,执行完成线程就会推出。如果我们需要一个机制,让线程能...
    无神阅读 139评论 0 2
  • 留胡须的男人,我天生有些抗拒,对这些男人没好感。 也许有些女人很喜欢,觉得留胡须的男人很有型,很帅,觉得很有男人味...
    码字好玩儿阅读 236评论 0 0