RunLoop-基础概念(初识篇)

学习这篇内容主要讲解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开启全局的一个子线程,运行代码后,在子线程会无限循环的一直在跑,不会停!这是因为:

  • 这个RunLoopModlesources为空、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.

你可能感兴趣的:(RunLoop-基础概念(初识篇))