学习这篇内容主要讲解RunLoop的概念,以及RunLoop和线程之间的关系。
当然提及RunLoop也离不开Autorealse Pool,本篇内容略有提及,但不重点阐述。
本篇内容是我自己对RunLoop概念的总结,和简单呈现,内容比较精炼。
概念
- RunLoop是系统中和线程相关的基础架构的组成部分,一个RunLoop是一个事件处理环,系统利用这个事件处理环来安排事务。
- RunLoop的意义是让你的线程在有工作的时候去干活,没有工作的时候进入休眠节省系统资源。
- 每个线程(包含主线程)都有一个Runloop。对于每一个Runloop,系统会隐式创建一个Autorelease pool,这样所有的Autorelease Pool会构成一个像callstack一样的一个栈式结构,在每一个Runloop结束时,当前栈顶的Autorelease pool会被销毁,这样这个Pool里的每个Object会被release。
- RunLoop和线程之间是以键值对应的形式一一对应的,其中key是thread,value是RunLoop,RunLoop也是管理线程的一种机制,这种机制不仅在iOS上有,在安卓的Looper,Node.js中的EventLoop都有类似的模式。
唤醒一个线程其实就是唤醒线程的source。
一、初识RunLoop
首先,我们在主线程中添加如下代码:
while (1) {
NSLog(@"while begin");
// the thread be blocked here
NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
[runLoop runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
// this will not be executed
NSLog(@"while end");
}
我们将上面代码在主线程运行,我们会发现while end
没有执行,过一会又执行了,这是因为:
- 主线程中,本身有自己的
RunLoop
,所以主线程可以一直不被释放,在需要做事情的时候主线程被唤醒干活, - 在不需要做事情的时候主线程会休眠,所以上面代码到
distantFuture
休眠线程后,会停止执行,当主线程需要处理某些事情的时候才会被唤醒。
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
while (1) {
NSLog(@"while begin");
NSRunLoop *subRunLoop = [NSRunLoop currentRunLoop];
[subRunLoop runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
NSLog(@"while end");
}
});
上面的代码是通过GCD
开启全局的一个子线程,运行代码后,在子线程会无限循环的一直在跑,不会停!这是因为:
- 这个
RunLoopModle
中sources
为空、observers
为空、timers
为空,所以这个RunLoop
直接就结束释放了. - 我们看到虽然有Mode,但是我们没有给它
soures,observer,timer
,其实Mode中的这些source,observer,timer
,统称为这个Mode的item,如果一个Mode中一个item都没有,则这个RunLoop会直接退出,不进入循环(其实线程之所以可以一直存在就是由于RunLoop
将其带入了这个循环中)。下面我们为这个RunLoop
添加个source:
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
while (1) {
NSPort *macPort = [NSPort port];
NSLog(@"while begin");
NSRunLoop *subRunLoop = [NSRunLoop currentRunLoop];
[subRunLoop addPort:macPort forMode:NSDefaultRunLoopMode];
[subRunLoop runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
NSLog(@"while end");
NSLog(@"%@",subRunLoop);
}
});
上面的代码,运行后,会停在休眠的那一行代码,因为我们给RunLoop的model添加item.
小结:我们的RunLoop
要想工作,必须要让它存在一个Item(source,observer或者timer
),主线程之所以能够一直存在,并且随时准备被唤醒就是应为系统为其添加了很多Item.