iOS知识梳理:RunLoop

相关连接:
深入理解RunLoop
IOS---实例化讲解RunLoop
iOS知识点整理-RunLoop
RunLoop的前世今生

RunLoop的概念

RunLoop是一个时间处理环,系统利用这个时间处理环来安排事物,协调输入的各种时间.RunLoop的目的是让你的线程在有工作的时候忙碌,没有工作的时候休眠(和线程相关)

一般来讲,一个线程只能执行一个任务,执行完成后线程就会退出.....但是实际情况是,我们希望有一个机制,,让线程能随时处理时间但并不退出....当我们给他发送退出指令是它才退出...

在iOS中RunLoop就是用来实现这种机制的...这种机制的关键点在于:如何处理时间和消息,让线程在没有处理消息的时候休眠以避免资源占用,在有消息到来时立刻被唤醒

RunLoop实际上就是一个对象...这个对象管理其需要处理的时间和消息, 并提供了一个入口函数来执行上诉逻辑....线程执行了这个函数后...就会一直处于这个函数内部 "接收消息 ->等待 -> 处理"的循环中,知道这个循环结束(比如传入quit消息),函数返回..

iOS提供了两个这样的对象 NSRunLoop和CFRunLoopRef

RunLoop和线程的关系

线程和RunLoop之间是一一对应的,其关系是保存在一个全局的Dictionary里....线程创建之后是没有RunLoop的(主线程除外),RunLoop的创建时发生在第一次获取时.

苹果不允许世界创建RunLoop, 但是可以通过[NSRunLoop currentRunLoop]或者CFRunLoopGetCurrent()来获取(如果没有就会自动创建一个)

RunLoop对外的接口

CoreFondation里关于RunLoop有4个类:
CFRunLoopRef
CFRunLoopModeRef
CFRunLoopSourceRef
CFRunLoopTimerRef
CFRunLoopObserverRef
RunLoop共包含5个类,但公开的只有Source,Timer,Observer相关的三个类.

1.RunLoop Modes
一个RunLoop包含若干个Mode,每个Mode又包含若干个Source/Timer/Observer每次调用RunLoop的函数是,只能指定其中一个mode,这个mode被称为CurrentMode....如果需要切换Mode..只能退出Loop, 再重新指定一个Mode进入.这样做是为了分割开不通组的Source/Timer/Observer,让其相互不影响.

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

2.RunLoop Timer
基于时间的触发器,基本上就是说NSTimer
在新建NSTimer之后,要把timer添加到RunLoop中,否则timer不会执行.

NSTimer *timer = [NSTimer timerWithTimerInterval:5.0 target:self selector:@selector(show) userInfo:nil repeats:YES];
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];

add到相应的Mode定时器的方法就只能在相应的Mode下才生效,
也可以把Mode设置为NSRunLoopCommonModes,就可以再默认模式和追踪模式下都能运行.
如果是通过scheduledTimerWithTimeInterval创建的NSTimer,默认添加到RunLoop的DefaultMode中,所以他会自动运行.

当其加入到RunLoop时,RunLoop会注册对应的时间点,当时间点到时,RunLoop会被唤醒以执行那个回调.如果线程阻塞或者不在这个Mode下,触发点将不会执行,一直等到下一个周期时间点的触发.

3.RunLoop Source
CFRunLoopSourceRef是事件源,比如外部的触摸,点击事件和系统内部进程间的通信等.
Source有两个版本:Source0和Source1
Source0:非基于Port的,只包含了一个回调,它并不能主动触发时间.使用时,你需要先调用CFRunLoopSourceSignal(source),讲这个Source标记为待处理,然后手动调用CFRunLoopWakeUp(runloop)来唤醒RunLoop.让其处理这个事件.
Source1:基于Port的,包含了一个mach_port和一个回调,被用于通过内核和其他线程相互发送消息.这种Source能主动唤醒RunLoop的线程,后面讲到AFNetworking创建的常驻线程就是在线程中添加一个NSport来实现的.

4.RunLoop Observer
CFRunLoopObserverRef是观察者,每个Observer都包含一个回调,当RunLoop的状态发生变化时,观察者就能通过回调接收到这个变化.

RunLoop的状态变化

3.RunLoop的运行机制

RunLoop的运行机制

RunLoop的实际应用

1.AutoreleasePool

app启动后,系统启动主线程并创建了RunLoop,在主线程里注册了两个observer,回调都是_wrapRunLoopWithAutoreleasePoolHandle()
第一个observer监听一个事件:即将进入Loop(kCFRunLoopEntry),
调用_objc_autoreleasePoolPush()创建一个栈自动释放池,这个优先级最高,保证创建释放池在其他操作之前
第二个observer监听两个事件:准备进入休眠(kCFRunLoopBeforeWaiting)和即将退出Loop(kCFRunLoopExit)
进入休眠释放旧的池并创建新的池,退出是释放掉自动释放池.

在主线程中执行代码一般都是写在事件回调或Timer回调中的,这些回调都被加入了主线程的自动释放池中,所以在ARC模式下我们不用关心对象什么时候释放,也不用去创建和管理pool.(如果事件不在主线程中要注意创建自动释放池,否则可能会出现内存泄漏)

2.NSTimer优化使用

开放中常见的现象,在界面上有个UIScrollView控件,如果此时还要一个定时器在执行一个事件,你会发现当你滚动scrollView的时候,定时器会失效..
因为timer用scheduledTimerWithTimeInterval:初始化的时候默认关联为DefaultMode,在主线程UITrackingRunLoopMode优先级最高,在用户拖动控件时,主线程的RunLoop运行在UITrackingRunLoopMode下,因此系统不会立即执行Default Mode下的事件.

解决方法1.把当前timer加入到UITrackingRunLoopMode或者kCFRunLoopCommonModes中.
解决方法2.使用不受RunLoop影响的GCD创建定时器.

3.滑动时加载图片,滑动不流畅的问题.

用户滑动scrollview的过程中加载图片,由于UI的操作都是在主线程进行的,会造成滑动不流畅的问题,这个时候我们就需要在滑动的不加载图片,等滑动操作完成在进行加载图片的操作.

UIImage *downloadImage = ...
[self.imageView performSelector:@selector(setImage:) 
                        withObject: downloadImage 
                        afterDelay:3.0 
                        inModes:@[NSDefaultRunLoopMode]];

讲setImage放到DefaultMode里确定UITrackingRunLoopMode下该操作不会被执行...

4.网络请求

AFN每次进行的网络操作,开始,暂停,取消操作时,都将相应的执行任务扔进了自己创建的线程RunLoop中进行处理,从而避免造成主线程的阻塞.

5.处理崩溃让程序继续运行

我们都知道,如果App运行遇到 Exception 就会直接崩溃并且退出,其实真正让应用退出的并不是产生的异常,而是当产生异常时,系统会结束掉当前主线程的 RunLoop ,RunLoop 退出主线程就退出了,所以应用才会退出。明白这个道理,去完成这个“不可能的任务”就很简单了。

CFRunLoopRef runLoop = CFRunLoopGetCurrent();
CFArrayRef allModes = CFRunLoopCopyAllModes(runLoop);
while (!isQuit){
    for (NSString *mode in (__bridge NSArray *)allModes) {
        CFRunLoopRunInMode((CFStringRef)mode, 0.001, false);
    }
}
CFRelease(allModes);

把上面的代码添加到 Exception 的handle方法中,此时创建了一个 RunLoop ,让这个 RunLoop 在所有的 Mode 下面一直不停的跑,保证主线程不会退出,我们的应用也就存活下来了。

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

推荐阅读更多精彩内容