iOS开发:RunLoop相关分析总结

什么是Runloop?
Runloop是通过内部维护的事件循环来对事件和消息进行管理的一种机制。当没有消息需要处理的时候,线程进入休眠以避免占用资源,有消息需要处理时,立即被唤醒。

runloop循环不是单独的do-while循环,而是发生一个用户态到内核态切换,以及内核态到用户态切换。它维护的事件循环可以用来不断的处理消息和事件,当没有消息和事件需要处理时会从用户态切换到内核态,由此可以用来休眠线程,避免资源占用。当有消息需要处理时会从内核态切换到用户态,当前线程会被唤醒,所以状态切换才是runloop的关键。

iOS中提供了两套Runloop接口,一个是NSRunLoop基于Objective-C,在Foundation框架中,另一个是CFRunLoopRef基于C,在CoreFoundation中。而NSRunLoop是对CFRunLoopRef的封装,两者接口基本都是对应的。CFRunLoopRef runloop = [nsrunloop getCFRunLoop]可以获取对应的CFRunLoopRef。通过一个表格来对比一下,后面我们将通过对CFRunloop的解读来更深刻的理解Runloop

特征 NSRunLoop CFRunLoopRef
所属框架 Objective-C/Foundation C/CoreFoundation
获取Runloop [NSRunLoop currentRunLoop]
[NSRunLoop mainRunLoop]
CFRunLoopGetCurrent()
CFRunLoopGetMain()
Source事件 addPort:forMode:
removePort:forMode:
CFRunLoopAddSource(...)
CFRunLoopRemoveSource(...)
Timer事件 addTimer:forMode: CFRunLoopAddTimer(...)
Observer事件 CFRunLoopAddObserver(...)
CFRunLoopRemoveObserver(...)
run run
runUntilDate:
runMode:beforeDate:
CFRunLoopRun()
CFRunLoopRunInMode(...)
CFRunLoopRunSpecific(...)

1. __CFRunLoop相关数据结构

struct __CFRunLoop {
    ...
    pthread_t _pthread;//对应的线程
    CFMutableSetRef _commonModes;
    CFMutableSetRef _commonModeItems;
    CFRunLoopModeRef _currentMode;
    CFMutableSetRef _modes;
    ...
};

__CFRunLooprunloop本身:typedef struct __CFRunLoop *CFRunLoopRef__CFRunLoop对应多个__CFRunLoopMode

struct __CFRunLoopMode {
    ...
    CFStringRef _name;
    CFMutableSetRef _sources0;//
    CFMutableSetRef _sources1;
    CFMutableArrayRef _observers;
    CFMutableArrayRef _timers;
    ...
};

__CFRunLoopMode是runloop的运行模式:typedef struct __CFRunLoopMode *CFRunLoopModeRef。每一个__CFRunLoopMode又包含多个_sources0、_sources1、_observers、_timers事件。_sources0:非基于Port的,也就是用户主动发出的事件。_sources1:基于Port的,也就是系统内部的消息事件。_observers:观察者。
_timers:定时器事件。
系统默认注册了5中类型的Mode

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

主线程默认运行在kCFRunLoopDefaultMode下,滑动scrollView,就变成了UITrackingRunLoopMode,手指离开又变成了kCFRunLoopDefaultMode

相关的类的成员变量与关系:


__CFRunLoop.png

如上图时Runloop中用到的基础结构,再对应关系方面:一个__CFRunLoop实例可以包含多个__CFRunLoopMode;一个__CFRunLoopMode又包含多个CFRunLoopSourceRef、CFRunLoopObserverRef、CFRunLoopTimerRef事件。一个Runloop要想跑起来,内部必须要有一个Mode,并且这个Mode里边必须包含一个Source/Observer/Timer事件。

CFRunLoop的状态:

名称 说明
kCFRunLoopEntry 即将进入runloop
kCFRunLoopBeforeTimers 即将处理timer事件
kCFRunLoopBeforeSources 即将处理source事件
kCFRunLoopBeforeWaiting 即将进入睡眠
kCFRunLoopAfterWaiting 被唤醒
kCFRunLoopExit runloop退出

2._CFRunLoop的创建、运行、退出

-创建

[NSRunLoop mainRunLoop]对应底层的CFRunLoopGetMain(),[NSRunLoop currentRunLoop]对应底层的CFRunLoopGetCurrent(),内部都是通过CF_EXPORT CFRunLoopRef _CFRunLoopGet0(pthread_t t)获取的runloop,分别传入pthread_main_thread_np()pthread_self()也就是主线程和当前线程的id。

CF_EXPORT CFRunLoopRef _CFRunLoopGet0(pthread_t t) {
    if (pthread_equal(t, kNilPthreadT)) {
        //如果外部传入无效的0,则将主线程ID赋值给t。
        t = pthread_main_thread_np();
    }
    __CFLock(&loopsLock);
    if (!__CFRunLoops) {
        //如果__CFRunLoops为空,则创建主线程对应的runloop
        __CFUnlock(&loopsLock);
        CFMutableDictionaryRef dict = CFDictionaryCreateMutable(kCFAllocatorSystemDefault, 0, NULL, &kCFTypeDictionaryValueCallBacks);
        //__CFRunLoopCreate做线程的初始化
        CFRunLoopRef mainLoop = __CFRunLoopCreate(pthread_main_thread_np()); 
        // 将mainLoop保存到dict中,以线程id为key,mainLoop为value
        CFDictionarySetValue(dict, pthreadPointer(pthread_main_thread_np()), mainLoop);
        //将dict中的内容复制到__CFRunLoops地址上
        if (!OSAtomicCompareAndSwapPtrBarrier(NULL, dict, (void * volatile *)&__CFRunLoops)) {
            CFRelease(dict);
        }
        CFRelease(mainLoop);
        __CFLock(&loopsLock);
    }
    //通过线程id获取runloop
    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) {
            //如果创建后还不能获取到则使用刚才创建的,并将newLoop保存到__CFRunLoops中
            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);
    }
    if (pthread_equal(t, pthread_self())) {
        //如果获取的是当前线程的
        _CFSetTSD(__CFTSDKeyRunLoop, (void *)loop, NULL);
        if (0 == _CFGetTSD(__CFTSDKeyRunLoopCntr)) {
            _CFSetTSD(__CFTSDKeyRunLoopCntr, (void *)(PTHREAD_DESTRUCTOR_ITERATIONS-1), (void (*)(void *))__CFFinalizeRunLoop);
        }
    }
    return loop;
}

__CFRunLoopCreate中通过_CFRuntimeCreateInstance实例华再进行其他变量的一些初始化。其中loop->_pthread = t;将线程id绑定到了runloop上。runloop通过线程id去查找,如果没有则进行创建并将线程id绑定到runloop上,通过这个规则我们知道:线程与runloop一一对应;线程不一定都有runloop,首个runloop创建时会检查__CFRunLoops是否为空,为空则先创建主线程的runloop,再创建指定线程的runloop

-运行

启动Runloop,调用CFRunLoopRun()即可,Runloop进入运行循环,运行状态只要不是kCFRunLoopRunStoppedkCFRunLoopRunFinished就会一直运行下去不退出。

void CFRunLoopRun(void) {    /* DOES CALLOUT */
    int32_t result;
    do {
        result = CFRunLoopRunSpecific(CFRunLoopGetCurrent(), kCFRunLoopDefaultMode, 1.0e10, false);
        CHECK_FOR_FORK();
    } while (kCFRunLoopRunStopped != result && kCFRunLoopRunFinished != result);
}

//将runloop的运行状态切换到指定的Mode

//代码较长,只列出重要步骤的
SInt32 CFRunLoopRunSpecific(CFRunLoopRef rl, CFStringRef modeName, CFTimeInterval seconds, Boolean returnAfterSourceHandled) {
    ...
    CFRunLoopModeRef currentMode = __CFRunLoopFindMode(rl, modeName, false);
    //如果没有获取到mode或者mode中的事件为空(无sources0/sources1等)返回kCFRunLoopRunFinished
    if (NULL == currentMode || __CFRunLoopModeIsEmpty(rl, currentMode, rl->_currentMode)) {
        return kCFRunLoopRunFinished;
    }
    //保存previousPerRun、previousMode
    volatile _per_run_data *previousPerRun = __CFRunLoopPushPerRunData(rl);
    CFRunLoopModeRef previousMode = rl->_currentMode;
    rl->_currentMode = currentMode;
    int32_t result = kCFRunLoopRunFinished;

    //通知Observer即将进入循环
    if (currentMode->_observerMask & kCFRunLoopEntry ) __CFRunLoopDoObservers(rl, currentMode, kCFRunLoopEntry);
    result = __CFRunLoopRun(rl, currentMode, seconds, returnAfterSourceHandled, previousMode);
    //通知Observer即将退出循环
    if (currentMode->_observerMask & kCFRunLoopExit ) __CFRunLoopDoObservers(rl, currentMode, kCFRunLoopExit);

    //恢复previousPerRun,previousMode
    __CFRunLoopPopPerRunData(rl, previousPerRun);
    rl->_currentMode = previousMode;
    return result;
}

3.Runloop的使用

3.1.main函数为何能保持不退出?

main函数中,会调用UIApplicationMain函数,在内部会启动主线程的Runloop,可以不断的接收消息,比如点击屏幕事件,滑动列表以及处理网络请求的返回等接收消息后对事件进行处理,处理完之后,就会继续等待。

3.2.NSTimer相关案例

案例1:

NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(run) userInfo:nil repeats:YES];

我们主线程执行如下代码,我们Timer能够正常运行,但是如果我们在进行scrollView滑动的时候定时器会停止,这是什么原因呢?在新开子线程调用它运行不起来,这是什么原因呢?

这里我们需要明白:1.scheduledTimerWithTimeInterval:方法会自动把当前初始化的Timer加入到currentRunLoopkCFRunLoopDefaultMode模式下,主线程的runloop已经在run状态了,所以定时器会立即启动。如果手动滑动scrollView,则主线程的runloop的状态切换为UITrackingRunLoopMode模式了,添加在kCFRunLoopDefaultMode模式的Timer自然就没有回调了。解决办法:将Timer手动添加到UITrackingRunLoopMode模式或者kCFRunLoopCommonModes模式即可。

如果在新开的子线程执行上面的代码,由于新开的子线程并不会主动创建runloop,所以定时器自然运行不起来。解决办法:手动将Timer加入到currentRunLoopkCFRunLoopCommonModes模式,并且执行run方法。

3.3监听runloop的运行状态
-(void)addObserver{
    /*1.创建监听者
      第一个参数:怎么分配存储空间
      第二个参数:要监听的状态 。kCFRunLoopAllActivities表示所有的状态
      第三个参数:是否持续监听
      第四个参数:优先级 总是传0
      第五个参数:当状态改变时候的回调
      */
    CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
        switch (activity) {
            case kCFRunLoopEntry:
                NSLog(@"即将进入runloop"); break;
            case kCFRunLoopBeforeTimers:
                NSLog(@"即将处理timer事件"); break;
            case kCFRunLoopBeforeSources:
                NSLog(@"即将处理source事件");break;
            case kCFRunLoopBeforeWaiting:
                NSLog(@"即将进入睡眠"); break;
            case kCFRunLoopAfterWaiting:
                NSLog(@"被唤醒"); break;
            case kCFRunLoopExit:
                NSLog(@"runloop退出");break;
            default:
                break;
        }
    });
    /*2.添加监听者
       第一个参数:要监听哪个runloop
       第二个参数:观察者
       第三个参数:运行模式
       */
    CFRunLoopAddObserver(CFRunLoopGetCurrent(), observer, kCFRunLoopDefaultMode);
}
3.4常驻线程

一个线程要想跑起来,则需要至少一个mode,一个事件。所以我们可以使用Timer或者Source事件。
第一步创建一个新的线程

- (void)createThread {
    self.thread = [[NSThread alloc] initWithTarget:self selector:@selector(task1) object:nil];
    [self.thread start];
}

第二步在新开的线程添加Timer或者Source事件。

- (void)task1{
  [[NSRunLoop currentRunLoop] addPort:[NSPort port] forMode:NSDefaultRunLoopMode];
  [[NSRunLoop currentRunLoop] run];
}

第三部测试线程是否正常运行。可以通过点击等连续触发,也可以在主线程多次调用test来测试

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

推荐阅读更多精彩内容