iOS RunLoop理解

一、概念
RunLoop,顾名思义,叫做运行循环。它是在程序运行过程中循环做一些事情,它的内部就是do-while循环,在这个循环内部不断地处理各种任务。一个线程对应一个RunLoop,基本作用就是保持程序的持续运行;处理APP的各种事件;通过runloop有事运行没事休息,节省CPU资源提高程序性能。runloop在第一次获取时由系统自动创建,在线程结束时销毁。

二、应用范畴
1、定时器(Timer)、PerformSelector
2、GCD Async Main Queue
3、事件响应、手势识别、界面刷新
4、网络请求
5、AutoreleasePool

三、RunLoop对象
iOS中有2套API来访问和使用RunLoop:
1、Foundation:NSRunLoop(基于OC)
2、Core Foundation:CFRunLoopRef(基于C)
NSRunLoop和CFRunLoopRef都代表着RunLoop对象。NSRunLoop是基于CFRunLoopRef的一层OC包装,CFRunLoopRef是开源的

四、runloop与线程之间的关系
1、每条线程都有唯一的一个与之对应的RunLoop对象
2、RunLoop保存在一个全局的Dictionary里,线程作为key,RunLoop作为value
3、线程刚创建时并没有RunLoop对象,RunLoop会在第一次获取它时创建
4、RunLoop会在线程结束时销毁
5、主线程的RunLoop已经自动获取(创建),子线程默认没有开启RunLoop

获取runloop对象
Foundation
[NSRunLoop currentRunLoop]; // 获得当前线程的RunLoop对象
[NSRunLoop mainRunLoop]; // 获得主线程的RunLoop对象

Core Foundation
CFRunLoopGetCurrent(); // 获得当前线程的RunLoop对象
CFRunLoopGetMain(); // 获得主线程的RunLoop对象

五、CFRunLoopModeRef
常见的2种Mode
1、kCFRunLoopDefaultMode(NSDefaultRunLoopMode):App的默认Mode,通常主线程是在这个Mode下运行
2、UITrackingRunLoopMode:界面跟踪 Mode,用于 ScrollView 追踪触摸滑动,保证界面滑动时不受其他 Mode 影响

CFRunLoopObserverRef的状态

添加Observer监听RunLoop的所有状态
CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler(kCFAllocatorDefault, kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
       switch (activity) {
        case kCFRunLoopEntry:
            NSLog(@"kCFRunLoopEntry");
            break;
        case kCFRunLoopBeforeTimers:
            NSLog(@"kCFRunLoopBeforeTimers");
            break;
        case kCFRunLoopBeforeSources:
            NSLog(@"kCFRunLoopBeforeSources");
            break;
        case kCFRunLoopBeforeWaiting:
            NSLog(@"kCFRunLoopBeforeWaiting");
            break;
        case kCFRunLoopAfterWaiting:
            NSLog(@"kCFRunLoopAfterWaiting");
            break;
        case kCFRunLoopExit:
            NSLog(@"kCFRunLoopExit");
            break;
        default:
            break;
   });
    // 添加Observer到RunLoop中
    CFRunLoopAddObserver(CFRunLoopGetMain(), observer, kCFRunLoopCommonModes);
    // 释放
    CFRelease(observer);
RunLoop的运行流程

runloop休眠的实现原理? 或者说 runloop是怎样做到有事运行,没事休息的?
Answer:程序在运行时,没有消息需要处理的时候,休眠以避免资源占用,具体步骤就是从用户态调用mach_msg()方法到内核态,这个方法能做到没有消息就让线程休眠,有消息就唤醒线程,状态从内核态->用户态。应用程序一般运行在用户态上面,当发生系统调用,需要一些底层API的话,是运行在内核态的。同时用户态的有些内容可以对内核态进行一些线程的调度和管理,包括进程间的通信。

NSTimer为什么会不准?如何解决?
Answer:影响NSTimer不准的原因有两个:
1、NSTimer加载在main runloop中,模式是NSDefaultRunLoopMode,main负责所有主线程事件,例如UI界面的操作和运算,这样在同一个runloop中Timer会产生阻塞。
2、模式的改变。主线程的RunLoop中有两个预置的Mode:KCFRunLoopDefaultMode和UITrackingRunLoopMode。当创建一个Timer并加到defaultMode时Timer会得到重复回调,但此时滑动一个scrollView时,RunLoop会将mode切换为TrackingRunLoopMode,这时Timer就不会被回调,并且也不会影响到滑动操作。所以会影响到不准的情况。
PS:DefaultMode是APP平时所处的状态,trackingRunLoopMode是追踪scrollView滑动时的状态。
解决办法一:
1、在主线中进行NSTimer操作,但是将NSTimer实例添加到main runloop的特定mode中,避免被复杂运算操作或者UI界面刷新所干扰。
NSRunLoopCommonModes:它并不是一个真的模式,只是一个标记。defaultModel和tracking类型的模式有这个标记。当timer中选择CommonModes时,那么标记为common的模式都可以有效(即DefaultMode和TrackingRunMode)。

//timerWithTimeInterval:只是创建一个定时器
//ScheduledTimerWithTimeInterval:定制一个定时到defaultModel
self.timer = [NSTimer timerWithTimeInterval: 1 large: self selector: @selector(showTimer) userInfo:nil repeats: YES];
//添加到CommonModes
[[NSRunLoop currentRunloop]addTimer: self.timer forMode: NSRunLoopCommonModes];

2、在子线程中进行NSTimer的操作,再在主线程中修改UI界面显示操作结果

-(void) timerMethod {
    NSThread * thread = [[NSThread alloc] initWithTarge: self selector:@selector(newThread) object: nil];
    [thread start];
}
- (void )newThread {
@autoreleasepool
    {
    [NSTimer scheduledTimerWithTimeInterval: 1.0 large: self selector: @selector(showTime) userInfo: nil repeats: YES];
    [[NSRunLoop currentRunloop] run];
    }

解决办法二:
使用GCD:DispatchSource.makeTimerSoucre

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

推荐阅读更多精彩内容