Runloop

Runloop

  • 多线程编程指南
  • 资料:
    • 1.开源网址中下载CF开头的包,CF是CoreFoundation的缩写,CFRnLoop.c是实现文件
    • 2.官方文档

Runloop与线程

  • 每条线程都有唯一的一个与之对应的Runloop对象
    • CFRunLoopGet0(pthread_t t)
  • 主线程的Runloop已经创建好了,子线程的Runloop需要主动创建
  • Runloop在第一次获取时创建,在线程结束的时候销毁

获得Runloop对象

  • Foundation(OC)
    • [NSRunloop currentRunloop]
    • [NSRunloop mainRunloop]
  • Core Foundation(C)
    • CFRunLoopRef
    • CFRunLoopGetCurrent();
    • CFRunLoopGetMain();
  • 转换OC-C
    • mainRunloop.getCFRunLoop
  • RunLoop和线程的关系 一一对应
    • [NSThread detachNewThreadSelector:toTarget:withObject:]
    • 主线程对应的RunLoop默认已经创建了,子线程对应的runloop需要我们手动创建
  • [NSRunloop currentRunloop]是懒加载的,只会初始化一次,会判断当前线程对应的runloop是否存在,如果存在,直接返回,如果不存在会创建一个,返回给我们,该方法就是用来创建Runloop的。

Runloop相关类

  • CoreFoundation中关于RunLoop的五个类:

    • CFRunLoopRef
    • CFRunLoopModeRef:运行模式
    • CFRunLoopSourceRef:事件源、输入源
    • CFRunLoopTimerRef:定时器事件
    • CFRunLoopObserverRef:监听者、观察者
  • 关系:

    • Runloop要启动必须要选择一种运行模式,有多种运行模式可供选择,但只能选择一种
    • 运行模式里面有source/observer/timer,runloop启动之前会检查运行模式是否为空(怎么判断是否为空?检查有没有source和timer,如果一个source和timer都没有就为空,如果为空,runloop启动之后就退出了)
    • 开启runloop运行循环-死循环
  • 运行模式:CFRunloopModeRef

    • CFRunLoopRef代表RunLoop的运行模式
    • 一个RunLoop包含若干个Mode,每个Mode又包含若干个Source/Timer/Observer
    • 每一次Runloop启动时,只能指定其中一个Mode,这个Mode被称作currentMode
    • 如果需要切换Mode,只能退出Loop,再重新指定一个Mode进入
    • 这样做主要是为了分隔开不同组的Source/Timer/Observer,让其互不影响

CFRunLoopModeRef

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

CFRunLoopModeRef和NSTimer

  • 定时器使用
    • 方法一:[NSTimer scheduledTimerWithTimerInterval:target:selector:userInfo:repeats:]
      • 该方法内部会自动将创建的定时器对象添加到当前的runloop当中,并且指定runloop的运行模式为默认
      • 开一个子线程调用这个方法,这个定时器能工作吗?
        • 不能工作!!!子线程对应的runloop没有创建
        • 手动创建runloop,手动添加定时器,还是不可以!!
  • 注意
    • 子线程对应的runloop需要手动的创建
    • 子线程创建的runloop还需要手动的开启 [currentRunLoop run]
    • 方法二:[NSTimer timerWithTimerInterval:target:selector:userInfo:repeats:]
      • 使用这个方法需要把定时器添加到runloop中
      • [NSRunloop currentRunloop] addTimer:forMode:
      • 问题:拖拽textField的时候,定时器不工作的原因?
        • 滚动textField,runloop运行模式改变了,切换到了UITrackingMode
      • 问题:怎么让定时器,在滚动textField的时候,也能工作?
        • 方法一:把定时器对象添加到runloop中,并且制定runloop的运行模式为默认,只有runloop的运行模式为NSDefaultRunloopMode的时候定时器才工作,要求在滚动textField的时候定时器也能工作,并且指定runloop的运行模式为tracking的时候,定时器才工作
        • 方法二:占位模式(等价于上面的两行代码)NSRunLoopCommonModes = Default & Tracking
          • NSRunLoopCommonModes表示把定时器对象添加到runloop中并且指定运行模式为Default或者是Tracking

GCD中的定时器(掌握)

  • 第一个函数:创建定时器对象
dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE__TIMER,0,0,dispatchQueue)
- 第一个参数:要创建的是一个定时器 
- 第二个参数:默认总是传0,描述信息
- 第三个参数:默认总是传0,
- 第四个参数:队列,并发队列(全局),决定代码块(dispatch_source_set_event_handler)在哪个线程中调用(主队列+主线程)
  • 第二个函数:设置定时器
dispatch_source_set_timer(timer,DISPATCH_TIME_NOW,intervalInSeconds *NSEC_PER_SEC,leewayInSeconds *NSEC_PER_SEC)
- 第一个参数:定时器对象
- 第二个参数:定时器开始计时的时间(开始时间)
    - 怎么修改开始时间?
        - dispatch_time_t t = dispatch_time(DISPATCH_TIME_NOW,延迟时间2.0*NSEC_PER_SEC)  
- 第三个参数:设置间隔时间 GCD的时间单位:纳秒
- 第四个参数:精准的,0表示绝对精准
  • 第三个函数:在block块里面要执行的任务,GCD定时器时间到了之后要执行的任务
dispatch_source_set_event_handler(timer,^{}) 
  • 第四个函数:恢复执行dispatch_resume(timer)

    • 默认是暂停状态,需要手动开启
  • 定时器不工作的原因:

    • GCD是不会受到runloop的影响
    • 原因是timer被释放了
      • 定一个属性,让timer不释放

RunloopSourceRef

  • CFRunloopSourceRef:是事件源(输入源)
  • 以前的分类:
    • Port-Based Source 基于端口
    • Custom Input Sources 自定义
    • Cocoa Perform Selector Sources
  • 现在的分类:
    • Source0:非基于Port
    • Source1:基于Port
    • 根据函数调用栈进行分类的

RunloopObserverRef

  • 01.创建一个监听者
    • CFRunloopObserverRef observer = CFRunLoopObserverCreateWithHandler()

      • 第一个参数,分配存储空间
        • CFAllocatorGetDefault()
      • 第二个参数:要监听的状态 0
      • 第三个参数:是否要持续监听YES
      • 第四个参数:和优先级相关 0
      • block块:当发现监听对象状态改变的时候调用该block
        • KCFRunLoopEntry:runloop进入启动
        • BeforeTimers:runloop即将处理timer事件
        • BeforeSources: runloop即将处理source事件
        • beforeWaiting:runloop即将进入睡眠状态
        • afterWaiting: runloop被唤醒
        • exit:runloop即将退出
        • AllActivities:监听所有的活动状态
    • 只能使用C语言代码

  • 02.设置监听
    • CFRunLoopAddObserver()
      • CFRunloopRef,要监听的runloop对象
      • CFRunLoopObserverRef,监听者对象本身
      • CFStringRef:runloop的运行模式
        • NSDefaultRunLoopMode(OC)
        • KCFRunLoopDefaultMode(C)✔️

runloop的运行流程

  • 处理逻辑(网友整理)
    • 1.通知内部Observer:即将进入Loop
    • 2.通知Observer:将要处理Timer
    • 3.通知Observer:将要处理Source0
    • 4.处理Source0
    • 5.如果有Source1,跳到第九步
    • 6.通知Observer:线程即将休眠
    • 7.休眠等待唤醒
    • 8.通知Observer:线程刚被唤醒
    • 9.处理唤醒时收到的消息,之后跳回2
    • 10.通知Observer:即将退出Runloop
  • 处理逻辑(官方)
    • Runloop的事件队列
    • 每次运行runloop,线程的runloop会自动处理之前未处理的消息,并通知相关的观察者,具体的顺序如下:
      • 1.通知观察者runloop已经启动
      • 2.通知观察者任何即将要开始的定时器
      • 3.通知观察者任何即将启动的非基于端口的源
      • 4.启动任何准备好的的非基于端口的源
      • 5.如果基于端口的源准备好并处于等待状态,立即启动,并进入步骤9
      • 6.通知观察者线程进入休眠
      • 7.将线程置于休眠直到下面的任意一个事件发生
        • 某一时间到达基于端口的源
        • 定时器启动
        • Runloop设置的时间已经超时
        • runloop被显示唤醒
      • 8.通知观察者线程将被唤醒
      • 9.处理未处理的事件
        • 如果用户定义的定时器启动,处理定时器事件并重启runloop,进入步骤2
        • 如果输入源启动,传递相应的消息
        • 如果runloop被显示唤醒而且时间还没超时,重启runloop,进入步骤2
      • 10.通知观察者runloop结束

代码模拟runloop死循环


void msg(int n)
{
    NSLog(@"runloop被唤醒");
    NSLog(@"runloop处理%zd事件",n);
}
int main(int argc, const char * argv[]) {
    @autoreleasepool {
       
        NSLog(@"runloop启动了");
        do {
            
            NSLog(@"runloop即将处理timer事件");
            NSLog(@"runloop即将处理source0事件");
            NSLog(@"source1事件");
            NSLog(@"runloop询问:还有事件需要我处理吗?");
            NSLog(@"runloop计入到休眠状态");
            
            int number = 0;
            scanf("%zd",&number);
            msg(number);
            
            
        } while (1);
    }
    return 0;
}

runloop的应用

  • NSTimer
  • ImageView显示
  • PerformSelector
  • selecter事件和runloop的关系
    • performSelector:withObject:afterDelay:inModes:
      • @[NSDefaultRunLoopMode,UITrackingRunLoopMode]
      • @[NSRunLoopCommonModes]
    • 默认情况下,selector事件是被添加到当前的runloop中执行的,并且指定了运行模式为默认
-(void)test1
{
     [NSThread detachNewThreadSelector:@selector(task) toTarget:self withObject:nil];
}

-(void)task
{
    [self.imageView performSelector:@selector(setImage:) withObject:[UIImage imageNamed:@"Snip20160713_9"] afterDelay:3.0];
    //运行模式启动之后,判断有一个selector事件,runloop才能够开启
    NSRunLoop *runloop = [NSRunLoop currentRunLoop];
    [runloop run];
    
    NSLog(@"+++++");
}
//需要进行线程间通信,否则会报错
  • 常驻线程
    • 开一条子线程,让子线程一直工作
    • 获得当前线程对应的runloop对象
    • 为runloop添加input source 或者是timer source
    • 启动runloop
//创建线程,执行任务
- (IBAction)createNewThreadBtnClick:(id)sender {
    
    //01 创建线程,执行任务
    //因为要拿到线程对象,所以用NSThread创建线程
    NSThread *thread = [[NSThread alloc]initWithTarget:self selector:@selector(run1) object:nil];
    
    //02 执行任务
    [thread start];
    
    self.thread = thread;
}


//继续执行任务
- (IBAction)goOnBtnClick:(id)sender {
    
    /*
    self.thread 任务执行完毕,已经进入到死亡状态|但是还没有被释放
    程序崩溃报错:attempt to start the thread again 
    [self.thread start];❎不能尝再次开启线程
    怎么让任务执行不完呢?在run1方法里搞一个死循环?这样做线程不会死,但是不会执行任务2
    正确做法:在run1方法里面开启一个runloop
    */
    
    //让之前创建的线程继续执行
    [self performSelector:@selector(run2) onThread:self.thread withObject:nil waitUntilDone:YES];
    
}

-(void)run1
{
    NSLog(@"run1---%@",[NSThread currentThread]);
    
    /*
    do {
        NSLog(@"-%@",[NSThread currentThread]);
    } while (1);
     */
    
    //001 获得当前线程对应的runloop对象
    NSRunLoop *currentRunloop = [NSRunLoop currentRunLoop];
    
    //002 为runloop添加input soucre或者是timer souce ,为了让运行模式不为空,否则不能开启runloop
    //基于端口的事件
    //好处:不需要调用方法,开发中常用
    [currentRunloop addPort:[NSPort port] forMode:NSDefaultRunLoopMode];
    
    //Perform事件
    //存在的问题:3.0s之后就退出了,开发中一般不用selector方法
    //[self performSelector:@selector(run3) withObject:nil afterDelay:3.0];
    //NSTimer事件
    //[NSTimer scheduledTimerWithTimeInterval:3.0 target:self selector:@selector(timerTest) userInfo:nil repeats:YES];
    
    //003 启动runloop
    //runUntilDate |run 内部都指定了运行模式为默认
    [currentRunloop run];
    //[currentRunloop runUntilDate:[NSDate dateWithTimeIntervalSinceNow:10000000]];
    
    NSLog(@"_______end____");//验证runloop是否开启了,打印(没有开启runloop)
    
}

-(void)run2
{
    NSLog(@"run2---%@",[NSThread currentThread]);
}
-(void)run3
{
    NSLog(@"run3---%@",[NSThread currentThread]);
}

-(void)timerTest
{
    NSLog(@"timer---");
}


runloop常见面试题

  • 什么是runloop?

    • 从字面意思看:运行循环、跑圈
    • 其实它内部就是do while循环,在这个循环内部不断地处理各种任务(比如Source/Timer/Observer)
    • 一个线程对应一个Runloop,主线程的runloop默认已经启动,子线程的runloop需要手动启动(调用run方法)
    • Runloop只能选择一个mode启动,如果当前mode中没有任何source(source0/source1)/timer,那么就直接退出runloop
  • 自动释放池什么时候释放?

    • 通过observer监听runloop的状态
    • 自动释放池的创建和释放
      • 创建:runloop启动的时候
      • 销毁:runloop退出的时候
      • 其他时候的创建和销毁:当runloop即将进入到休眠的时候会把之前的释放池销毁,重新创建一个新的自动释放池
  • 在开发中如何使用runloop?应用场景?

    • 开启一个常驻线程(让一个子线程不进入消亡状态,等待其他线程发来消息,处理其他事件)
      • 在子线程中开启一个定时器
      • 在子线程中进行一些长期监控
    • 可以控制定时器在特定模式下执行
    • 可以让某些事件(行为、任务)在特定模式下执行
    • 可以添加Observer监听Runloop的状态,比如监听点击事件的处理(在所有点击事件之前做一些事情)

总结:

  • 多图下载

    • 缓存策略:内存缓存+磁盘缓存
    • 开子线程下载
  • SDWebImage

    • 基本使用
    • 结构
      • caches
      • download
      • manager
    • 细节
  • Runloop

    • 概念:运行循环(死循环)
    • 作用【3】
      • 保持程序的持续运行
      • 处理app中的各种事件
      • 提高性能
    • 与线程关系:一一对应(字典)
      • 线程与runloop是一一对应的关系(字典)
      • 主线程对应的runloop已经创建并且启动mainRunloop
      • 子线程对应的runloop需要手动创建并且开启
      • 退出:线程销毁
    • runloop的相关类
      • runloopRef runloop本身
      • runloopModeRef 运行模式[5]
        • 默认Default
        • 界面追踪Tracking
        • commonModes
      • runloopTimeRef
      • observer 监听者
      • sourceRef 事件源
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容