RunLoop底层学习

概述

int main(int argc, char * argv[]) {
  @autoreleasepool {
      return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
  }
}

这是iOS项目中都会有的main.h文件,它底层的伪代码大概如此:

int main(int argc, char * argv[]) {
  @autoreleasepool {  
      int retVal = 0;
      do {
          //睡眠中等待消息
          int message = sleep_and_wait();
          //处理消息
          retVal = process_message(message);
      } while (0 == retVal);
      return 0;
  }
}
  • RunLoop,运行循环,在程序运行过程中,能保证循环做一些事情。
  • runloop和线程的关系
    程序的运行都是在线程中进行的,每条线程都有唯一的一个与之对应的RunLoop对象。为了保证这种唯一性,RunLoop保存在一个全局的Dictionary里,线程作为key值,RunLoop作为value值。
    CFRunLoopRef loop = (CFRunLoopRef)CFDictionaryGetValue(__CFRunLoops, pthreadPointer(t));
    __CFUnlock(&loopsLock);
    if (!loop) {
    CFRunLoopRef newLoop = __CFRunLoopCreate(t);
        __CFLock(&loopsLock);
    loop = (CFRunLoopRef)CFDictionaryGetValue(__CFRunLoops, pthreadPointer(t));
    if (!loop) {
        CFDictionarySetValue(__CFRunLoops, pthreadPointer(t), newLoop);
        loop = newLoop;
    }
        // don't release run loops inside the loopsLock, because CFRunLoopDeallocate may end up taking it
        __CFUnlock(&loopsLock);
    CFRelease(newLoop);
    }

线程刚创建时并没有RunLoop对象,RunLoop会在第一次获取它时创建;
RunLoop会在线程结束时销毁;
主线程的RunLoop已经自动获?。ù唇ǎ酉叱棠厦挥锌鬜unLoop。

  • 应用:RunLoop的应用范围很广,在定时器、GCD、performSelector、事件响应、手势识别、界面刷新、网络请求、AutoreleasePool等等很多地方用到。

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

RunLoop五个类和结构关系

1、CFRunLoopRef

struct __CFRunLoop {
    ...
    pthread_t _pthread;
    CFMutableSetRef _commonModes;
    CFMutableSetRef _commonModeItems;
    CFRunLoopModeRef _currentMode;
    CFMutableSetRef _modes;
    ...
};

2、CFRunLoopModeRef

typedef struct __CFRunLoopMode *CFRunLoopModeRef;
struct __CFRunLoopMode {
     ...
    CFStringRef _name;
     ...
    CFMutableSetRef _sources0;
    CFMutableSetRef _sources1;
    CFMutableArrayRef _observers;
    CFMutableArrayRef _timers;
    ...
};

3、CFRunLoopSourceRef;
4、CFRunLoopObserverRef;
5、CFRunLoopTimerRef。
从上面的图中可以看出,CFRunLoopRef中包含CFRunLoopModeRef,CFRunLoopModeRef包含_source、_observer、_timer。

  • 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会立马退出。

常见的两种Mode

  • kCFRunLoopDefaultMode( NSDefaultRunLoopMode):APP的默认Mode,通常主线程是在这个Mode下运行

  • UITrackingRunLoopMode:界面追踪Mode,用于ScrollView追踪触摸滑动,保证界面滑动时不受其他Mode影响

  • Source0
    触摸事件处理
    performSelector:onThread:
    点击处理

  • Source1
    基于Port的线程间通信
    系统事件捕捉

  • Timers
    NSTimer
    performSelector:withObject:afterDelay:

  • Observers
    用于监听RunLoop的状态
    UI刷新(BeforeWaiting)
    Autorelease pool(BeforeWaiting)

RunLoop运行逻辑

Int32 CFRunLoopRunSpecific(CFRunLoopRef rl, CFStringRef modeName, CFTimeInterval seconds, Boolean returnAfterSourceHandled) {  
     ......
         //通知Observers,进入Loop
     __CFRunLoopDoObservers(rl, currentMode, kCFRunLoopEntry);
        //具体要做的事
    result = __CFRunLoopRun(rl, currentMode, seconds, returnAfterSourceHandled, previousMode);
     //通知Observers,退出Loop
    __CFRunLoopDoObservers(rl, currentMode, kCFRunLoopExit);
    ......
    return result;
}
static int32_t __CFRunLoopRun(CFRunLoopRef rl, CFRunLoopModeRef rlm, CFTimeInterval seconds, Boolean stopAfterHandle, CFRunLoopModeRef previousMode) {
    .......
    do {
    ........
       //通知Observers:即将处理Timers
        if (rlm->_observerMask & kCFRunLoopBeforeTimers) __CFRunLoopDoObservers(rl, rlm, kCFRunLoopBeforeTimers);
        //通知Observers:即将处理Sources
        if (rlm->_observerMask & kCFRunLoopBeforeSources) __CFRunLoopDoObservers(rl, rlm, kCFRunLoopBeforeSources);
         //处理Block
        __CFRunLoopDoBlocks(rl, rlm);
        //处理Sources0
        Boolean sourceHandledThisLoop = __CFRunLoopDoSources0(rl, rlm, stopAfterHandle);
        if (sourceHandledThisLoop) {
            //处理Block
            __CFRunLoopDoBlocks(rl, rlm);
       }
        Boolean poll = sourceHandledThisLoop || (0ULL == timeout_context->termTSR);
          //判断有无Source1
        if (__CFRunLoopServiceMachPort(dispatchPort, &msg, sizeof(msg_buffer), &livePort, 0, &voucherState, NULL)) {
            //如果有Source1,就跳转到handle_msg
            goto handle_msg;
        }
        didDispatchPortLastTime = false;
       //通知Observers:即将休眠
       __CFRunLoopDoObservers(rl, rlm, kCFRunLoopBeforeWaiting);
        __CFRunLoopSetSleeping(rl);

        CFAbsoluteTime sleepStart = poll ? 0.0 : CFAbsoluteTimeGetCurrent();
        .......
        //等待别的消息来唤醒当前线程
       __CFRunLoopServiceMachPort(waitSet, &msg, sizeof(msg_buffer), &livePort, poll ? 0 : TIMEOUT_INFINITY, &voucherState, &voucherCopy);
       ......
      __CFRunLoopUnsetSleeping(rl);
        //通知Observers:结束休眠
      __CFRunLoopDoObservers(rl, rlm, kCFRunLoopAfterWaiting);

        handle_msg:;
        ......
        //被Timers唤醒
       if (modeQueuePort != MACH_PORT_NULL && livePort == modeQueuePort) {
            CFRUNLOOP_WAKEUP_FOR_TIMER();
            __CFRunLoopDoTimers(rl, rlm, mach_absolute_time())
        } else if (livePort == dispatchPort) {//被GCD唤醒
        .......
            __CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__(msg);
        .......
        } else {
        //处理Source1
         __CFRunLoopDoSource1(rl, rlm, rls, msg, msg->msgh_size, &reply) || sourceHandledThisLoop;
    }
     //处理Block
    __CFRunLoopDoBlocks(rl, rlm);
        //设置返回值
    if (sourceHandledThisLoop && stopAfterHandle) {
        retVal = kCFRunLoopRunHandledSource;
        } else if (timeout_context->termTSR < mach_absolute_time()) {
            retVal = kCFRunLoopRunTimedOut;
    } else if (__CFRunLoopIsStopped(rl)) {
            __CFRunLoopUnsetStopped(rl);
        retVal = kCFRunLoopRunStopped;
    } else if (rlm->_stopped) {
        rlm->_stopped = false;
        retVal = kCFRunLoopRunStopped;
    } else if (__CFRunLoopModeIsEmpty(rl, rlm, previousMode)) {
        retVal = kCFRunLoopRunFinished;
    }
        voucher_mach_msg_revert(voucherState);
        os_release(voucherCopy);

    } while (0 == retVal);
    ......
    return retVal;
}
  • 1、通知Observers监听器,即将进入Loop;
  • 2、通知Observers监听器,即将处理Timers;
  • 3、通知Observers监听器,即将处理Sources;
  • 4、处理被加入的block;
  • 5、处理Source0事件。这个事件被处理后,会在一个循环结束后被使用;
  • 6、判断是否有Source1,如果有,则跳转到handle_msg标签处执行;
  • 7、通知Observers,开始休眠;
  • 8、通知Observers,结束休眠,也是handle_msg标签处
    ??处理Timer
    ??处理GCD Async To Main Queue
    ??处理Source1
  • 9、处理blocks;
  • 10、根据前面的执行结果,决定如何操作
    ??是继续循环
    ??还是退出Loop
  • 11、通知Observers,退出Loop。

RunLoop的休眠原理

RunLoop的休眠原理

操作系统提供了两个API

  • 内核层面的API
  • 应用层面的API

当API在应用界面(用户态)时,调用mach_msg(),就会转到内核态,然后调用内核态的mach_msg(),当没有消息时,就线程休眠,有消息就唤醒线程。

RunLoop面试题

1、讲讲Runloop,项目中有用到吗
??控制线程生命周期(线程?;睿?br> AFNetworking使用RunLoop技术控制子线程的生命周期,让子线程一直在内存中,当需要接口请求是,唤醒AFNetworking。不停的创建子线程销毁子线程会影响性能。
??解决NSTimer在滑动时停止工作的问题
??监控应用卡顿
??性能优化
2、runloop内部实现逻辑
参考RunLoop运行逻辑
3、runloop和线程的关系
每条线程都有唯一的一个与之对应的RunLoop对象
RunLoop保存在一个全局的Dictionary里,线程作为key,RunLoop作为value
线程刚创建时并没有RunLoop对象,RunLoop会在第一次获取它时创建
RunLoop会在线程结束时销毁
主线程的RunLoop已经自动获?。ù唇ǎ酉叱棠厦挥锌鬜unLoop
4、timer与runloop的关系
RunLoop里面有个CFMutableSetRef _modes。这个模式里面有个CFMutableArrayRef _timers;
5、程序中添加每3秒响应一次的NSTimer,当拖动tableview时timer可能无法响应要怎么解决?
原因:定时器是在UITrackingRunLoopMode下工作,当拖动时,模式切换到kCFRunLoopDefaultMode模式下,定时器就停止工作。
解决办法:创建定时器,添加NSDefaultRunLoopMode和UITrackingRunLoopMode模式
6、runloop是怎么响应用户操作的,具体流程是什么样的
Source1捕捉用户操作,然后把这个事件包装成事件队列EventQueue,然后放到source0中处理
7、说说runloop的几种状态
kCFRunLoopEntry:即将进入Loop
kCFRunLoopBeforeTimers:即将处理Timers
kCFRunLoopBeforeSources:即将处理Source
kCFRunLoopBeforeWaiting:即将进入休眠
kCFRunLoopAfterWaiting:刚从休眠中唤醒
kCFRunLoopExit:即将退出Loop
8、runloop的mode作用是什么
常见的2中Mode
kCFRunLoopDefaultMode( NSDefaultRunLoopMode):APP的默认Mode,通常主线程是在这个
Mode下运行
UITrackingRunLoopMode:界面追踪Mode,用于ScrollView追踪触摸滑动,保证界面滑动时不受其他Mode影响
9、RunLoop自动释放池
1、自动释放池第一次创建:当runloop启动的时候
2、自动释放池最后一次销毁:当runloop退出的时候
3、自动释放池其他时间的创建和销毁:当runloop即将进入都休眠的时候,会把之前的自动释放池释放,重新创建一个新的自动释放池。

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

推荐阅读更多精彩内容