欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

RunLoop-基础概念(初识篇)

程序员文章站 2024-02-04 14:10:40
...

学习这篇内容主要讲解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.

转载于:https://www.jianshu.com/p/b5a3f981e1be