OC底层知识(十) : RunLoop

一、Runloop的简单介绍

Runloop 是在程序运行过程中循环做一些事情。比如应用在:定时器(NSTimer)、PerformSelector;GCD Async Main Queue;事件响应、手势识别、界面刷新、网络请求、AutoreleasePool

  • RunLoop的基本作用:保持程序的持续运行;处理App中的各种事件(如触摸事件、定时器事件等);节省CPU资源,提高程序性能:该做事时做事,该休息时休息
二、RunLoop对象的获取
  • 2.1、iOS中有2套API来访问和使用RunLoop

    • Foundation:NSRunLoop
    • Core Foundation:CFRunLoopRef
  • 2.2、NSRunLoop和CFRunLoopRef都代表着RunLoop对象

    NSRunLoop是基于CFRunLoopRef的一层OC包装

    CFRunLoopRef是开源的

  • 2.3、RunLoop与线程

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

    • OC(Foundation)获取RunLoop对象

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

      CFRunLoopGetCurrent(); // 获得当前线程的RunLoop对象
      CFRunLoopGetMain(); // 获得主线程的RunLoop对象
      
三、RunLoop相关的类

Core Foundation中关于RunLoop的5个类
CFRunLoopRef
CFRunLoopModeRef
CFRunLoopSourceRef
CFRunLoopTimerRef
CFRunLoopObserverRef

struct __CFRunLoop {
    CFRuntimeBase _base;
    pthread_mutex_t _lock;                 /* locked for accessing mode list */
    __CFPort _wakeUpPort;                  // used for CFRunLoopWakeUp 
    Boolean _unused;
    volatile _per_run_data *_perRunData;   // reset for runs of the run loop
    pthread_t _pthread;
    uint32_t _winthread;
    CFMutableSetRef _commonModes;
    CFMutableSetRef _commonModeItems;
    CFRunLoopModeRef _currentMode;
    CFMutableSetRef _modes;
    struct _block_item *_blocks_head;
    struct _block_item *_blocks_tail;
    CFAbsoluteTime _runTime;
    CFAbsoluteTime _sleepTime;
    CFTypeRef _counterpart;
};
  • 3.1、CFRunLoopModeRef

    • CFRunLoopModeRef代表RunLoop的运行模式
    • 一个RunLoop包含若干个Mode,每个Mode又包含若干个Source0/Source1/Timer/Observer



    • RunLoop启动时只能选择其中一个Mode,作为currentMode
    • 如果需要切换Mode,只能退出当前Loop,再重新选择一个Mode进入
      • 不同组的Source0/Source1/Timer/Observer能分隔开来,互不影响
    • 如果Mode里没有任何Source0/Source1/Timer/Observer,RunLoop会立马退出
    • CFRunLoopModeRef 2种Mode
      • kCFRunLoopDefaultMode(NSDefaultRunLoopMode):App的默认Mode,通常主线程是在这个Mode下运行
      • UITrackingRunLoopMode:界面跟踪 Mode,用于 ScrollView 追踪触摸滑动,保证界面滑动时不受其他 Mode 影响
        WechatIMG100.jpeg
  • 3.2、RunLoop的运行逻辑


    RunLoop的运行逻辑
    • Source0
      触摸事件处理
      performSelector:onThread:
    • Source1
      基于Port的线程间通信
      系统事件捕捉
    • Timers
      NSTimer
      performSelector:withObject:afterDelay:
    • Observers
      用于监听RunLoop的状态
      UI刷新(BeforeWaiting)
      Autorelease pool(BeforeWaiting)
RunLoop的运行逻辑
  • 3.3、RunLoop休眠的实现原理


    RunLoop休眠的实现原理
四、RunLoop在实际开中的应用 下面练习的demo
  • 4.1、控制线程生命周期(线程保活)

    • (1)OC核心代码实现保活 : 看demo里面的类JKCPermenantThread

      self.innerThread = [[MJThread alloc] initWithBlock:^{
       
       // 线程保活
       // 往RunLoop里面添加 Source\Timer\Observer
       [[NSRunLoop currentRunLoop] addPort:[[NSPort alloc] init] forMode:NSDefaultRunLoopMode];
       
        while (weakSelf && !weakSelf.isStopped) {
           [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
        }
      }];
      
    • (2)C核心代码实现保活:看demo里面的类ThreadCViewController

      self.innerThread = [[JKThread alloc] initWithBlock:^{
         NSLog(@"begin----");
       
         // 创建上下文(要初始化一下结构体)
         CFRunLoopSourceContext context = {0};
       
         // 创建source
         CFRunLoopSourceRef source = CFRunLoopSourceCreate(kCFAllocatorDefault, 0, &context);
       
         // 往Runloop中添加source
         CFRunLoopAddSource(CFRunLoopGetCurrent(), source, kCFRunLoopDefaultMode);
       
         // 销毁source
         CFRelease(source);
       
         // 启动
         CFRunLoopRunInMode(kCFRunLoopDefaultMode, 1.0e10, false);
       
         //            while (weakSelf && !weakSelf.isStopped) {
         //                // 第3个参数:returnAfterSourceHandled,设置为true,代表执行完source后就会退出当前loop
         //                CFRunLoopRunInMode(kCFRunLoopDefaultMode, 1.0e10, true);
         //            }
       
         NSLog(@"end----");
       }];
      
  • 4.2、解决NSTimer在滑动时停止工作的问题(我在做一个cell上动画的时候用了NSRunLoop来防止tableview滑动的时候cell上的动画暂停)

    NSTimer  *timer = [NSTimer scheduledTimerWithTimeInterval:0.1 target:self selector:@selector(clickTimeIndexPath) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
    /*
      UITrackingRunLoopMode和NSDefaultRunLoopMode是真正存在的模式
      [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
      [[NSRunLoop currentRunLoop] addTimer:timer forMode:UITrackingRunLoopMode];
      NSRunLoopCommonModes:n并不是一个真的模式,它只是一个标记
      timer 能在 _commandModes 数组中存放的模式下工作
      [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
     */
    
  • 4.3、监控应用卡顿(看底层七)

  • 4.4、性能优化(看底层七)

五、RunLoop的几个面试题
  • 5.1、讲讲 RunLoop,项目中有用到吗?(你可以回答里面的任意一个)
    答:我用到了:解决NSTimer在滑动时停止工作的问题(我在做一个cell上动画的时候用了NSRunLoop来防止tableview滑动的时候cell上的动画暂停)

  • 5.2、runloop内部实现逻辑?
    答:看上面的 3.2、RunLoop的运行逻辑

  • 5.3、runloop和线程的关系?
    答:看上面的 2.3、RunLoop与线程

  • 5.4、timer 与 runloop 的关系?
    答: timer 是运行在runloop里面的,runloop控制 timer什么时候执行

  • 5.5、程序中添加每3秒响应一次的NSTimer,当拖动tableview时timer可能无法响应要怎么解决?
    答:看上面的4.2,也就是要添加 NSRunLoopCommonModes标记

  • 5.6、runloop 是怎么响应用户操作的, 具体流程是什么样的?
    答: 由 source1捕获触摸或者响应事件,再由 source0去处理触摸或者响应事件

  • 5.7、说说runLoop的几种状态


    runLoop的几种状态
  • 5.8、runloop的mode作用是什么?
    答:runloop的mode 常用的有两种mode: NSDefaultRunLoopModeUITrackingRunLoopMode,在某一种mode下执行自己的source0、source1、timer,不同mode下互不影响,分工明确,使用起来更灵活、流畅。

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