RunLoop的使用总结

1、介绍RunLoop

什么是RunLoop

//runloop从字面的意思来看就是:跑 - 圈 也就是运行循环。
//以下是runloop的伪代码。。。
int main(int argc, char * argv[]) {
    BOOL running = YES;
    do {
        //执行各种任务,处理事件
    }while (running);
        return 0;
}

基本作用:

  • 保持程序的持续运行,如果没有RunLoop,程序执行完main函数就结束了。
  • 处理App中的各种事件(比如触摸事件、定时器事件、Selector事件)
  • 节省CPU资源,提高程序性能:该做事时做事,该休息时休息......

2、RunLoop对象

  • iOS中有2套API来访问和使用RunLoop
    Foundation框架中的NSRunLoop;
    Core Foundation中的CFRunLoop;

NSRunLoop是基于CFRunLoopRef的一层OC包装,所以要了解RunLoop内部结构,需要多研究CFRunLoopRef层面的API(Core Foundation层面)

3、RunLoop与线程

每条线程都有唯一的一个与之对应的RunLoop对象
主线程中的RunLoop由系统自动创建,子线程中RunLoop可以通过手动创建
RunLoop在线程结束的时候会被销毁

1、获取RunLoop对象Foundation框架中
[NSRunLoop currentRunLoop]; // 获得当前线程的RunLoop对象
[NSRunLoop mainRunLoop]; // 获得主线程的RunLoop对象

2、Core Foundation框架中
CFRunLoopGetCurrent(); // 获得当前线程的RunLoop对象
CFRunLoopGetMain(); // 获得主线程的RunLoop对象

3.注意点:开一个子线程创建runloop,不是通过alloc init方法创建,而是直接通过调用currentRunLoop方法来创建,它本身是一个懒加载的。
4.在子线程中,如果不主动获取Runloop的话,那么子线程内部是不会创建Runloop的。可以下载CFRunloopRef的源码,搜索_CFRunloopGet0,查看代码。
5.Runloop对象是利用字典来进行存储,而且key是对应的线程Value为该线程对应的Runloop。
4、RunLoop的结构
RunLoop相关联的类有五个
  • CFRunLoopRef
  • CFRunLoopSourceRef
  • CFRunLoopObserverRef
  • CFRunLoopTimerRef
  • CFRunLoopModeRef
runloop

RunLoop的结构
5、RunLoop相关类的理解

1.从上图可以看出

  • 一个RunLoop 可以有多个Mode(模式),每个模式又包含若干个 Source/Observer/Timer
  • RunLoop 一次只能运行一种Mode,切换Mode需要退出当前Mode

2.CFRunLoopModeRef 一共有五种模式

  • kCFRunLoopDefaultMode 默认模式,通常主线程在这个模式下运行
  • kCFRunLoopCommonModes 占位符(表示)
  • UITrackingRunLoopMode 界面跟踪Mode,用于追踪Scrollview的触摸滑动
  • UIInitializationRunLoopMode:刚启动App时进入的第一个Mode,启动后不在使用。
  • GSEventReceiveRunLoop:内部Mode,接收系事件。

3.CFRunLoopSourceRef(函数调用栈)

  • Source0:非基于Port(处理事件)
  • Source1:基于Port (接收分发系统事件)

4.CFRunLoopTimerRef:基于时间的触发器(和NSTimer差不多)
5.CFRunLoopObserverRef:监听RunLoop状态的观察者

CFRunLoopObserverRef是RunLoop的观察者,可以通过CFRunLoopObserverRef来监听RunLoop状态的改变

typedef CF_OPTIONS(CFOptionFlags, CFRunLoopActivity) {
      /* Run Loop Observer Activities */
      typedef CF_OPTIONS(CFOptionFlags, CFRunLoopActivity) {
          kCFRunLoopEntry = (1UL << 0),         // 状态值:1,表示进入RunLoop
          kCFRunLoopBeforeTimers = (1UL << 1),  // 状态值:2,表示即将处理NSTimer
          kCFRunLoopBeforeSources = (1UL << 2), // 状态值:4,表示即将处理Sources
          kCFRunLoopBeforeWaiting = (1UL << 5), // 状态值:32,表示即将休眠
          kCFRunLoopAfterWaiting = (1UL << 6),  // 状态值:64,表示从休眠中唤醒
          kCFRunLoopExit = (1UL << 7),          // 状态值:128,表示退出RunLoop
          kCFRunLoopAllActivities = 0x0FFFFFFFU // 表示监听上面所有的状态
      };
};
  • 如何监听RunLoop的状态:
  • 1.创建CFRunLoopObserverRef
// 第一个参数用于分配该observer对象的内存
// 第二个参数用以设置该observer所要关注的的事件,详见回调函数myRunLoopObserver中注释
// 第三个参数用于标识该observer是在第一次进入run loop时执行还是每次进入run loop处理时均执行
// 第四个参数用于设置该observer的优先级,一般为0
// 第五个参数用于设置该observer的回调函数
// 第六个参数observer的运行状态   
CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
      NSLog(@"----监听到RunLoop状态发生改变---%zd", activity);
      });
  • 2.将观察者CFRunLoopObserverRef添加到RunLoop上面
CFRunLoopAddObserver(CFRunLoopGetCurrent(), observer, kCFRunLoopDefaultMode);
  • 3.观察者CFRunLoopObserverRef要手动释放
 CFRelease(observer);````

6、RunLoop 处理逻辑

![RunLoop 处理逻辑](http://upload-images.jianshu.io/upload_images/785453-846c6532a9e0b7ba.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

#####6、RunLoop 使用
1.图片刷新(假如界面要刷新N多图片(渲染),此时用户拖拽UI控件就会出现卡的效果,我们可以通过RunLoop实现,只在RunLoop默认Mode下下载,也就是拖拽Mode下不刷新图片)
  • (void)viewDidLoad {
    [super viewDidLoad];
    // 只在NSDefaultRunLoopMode下执行(刷新图片)
    [self.myImageView performSelector:@selector(setImage:) withObject:[UIImage imageNamed:@"0"] afterDelay:2.0 inModes:@[NSDefaultRunLoopMode]];
    }

2.在RunLoop中创建定时器:
> - 1、第一种方式
  - 定时器NSTimer的创建方式一:timerWithTimeInterval:
 - 创建timer
 - 将timer 添加到RunLoop中
- 第二种方式
  NSTimer 和 scheduledTimerWithTimeInterval,这个不用加在runloop中,默认Mode:NSDefaultRunLoopMode

objc
// 创建一个定时器
NSTimer *timer = [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];
// NSDefaultRunLoopMode:NSTimer只有在默认模式下(NSDefaultRunLoopMode)工作,切换到其他模式不再工作,比如拖拽了界面上的某个控件(会切换成UITrackingRunLoopMode)
[[NSRunLoop mainRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
objc
// 创建一个定时器
NSTimer *timer = [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];
// 拖拽UI界面时出发定时器,在默认模式(NSDefaultRunLoopMode)下不工作
[[NSRunLoop mainRunLoop] addTimer:timer forMode:UITrackingRunLoopMode];
objc
// 创建一个定时器
NSTimer *timer = [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];
// NSRunLoopCommonModes仅仅是标记NSTimer在两种模式(UITrackingRunLoopMode/NSDefaultRunLoopMode)下都能运行,但一个RunLoop中同一时间内只能运行一种模式
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
objc
// 默认已经添加到主线程中RunLoop(mainRunLoop)中(Mode:NSDefaultRunLoopMode)
[NSTimer scheduledTimerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];


3.保证一个线程永远不死

import "ViewController.h"

/*
思路:为了保证线程不死,我们考虑在子线程中加入RunLoop,
但是由于RunLoop中没有没有源,就会自动退出RunLoop,
所以我们要为子线程添加一个RunLoop,
并且为这个RunLoop添加源(保证RunLoop不退出)
*/
@interface ViewController ()

/** 线程对象 */
@property (nonatomic, strong)NSThread *thread;

@end

@implementation ViewController

  • (void)viewDidLoad {
    [super viewDidLoad];

    // 创建子线程
    self.thread = [[NSThread alloc] initWithTarget:self selector:@selector(run) object:nil];

    //启动子线程
    [self.thread start];

}

  • (void)run {

    NSLog(@"run--%@", [NSThread currentThread]); // 子线程

    // 给子线程添加一个RunLoop,并且加入源
    [[NSRunLoop currentRunLoop] addPort:[NSPort port] forMode:NSDefaultRunLoopMode];
    // 启动RunLoop
    [[NSRunLoop currentRunLoop] run];
    NSLog(@"------------"); // RunLoop启动,这句没有执行的机会
    }

  • (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
    // 在子线程中调用test方法,如果子线程还在就能够调用成功
    [self performSelector:@selector(test) onThread:self.thread withObject:nil waitUntilDone:YES modes:@[NSDefaultRunLoopMode]];
    }

  • (void)test {
    NSLog(@"test--%@", [NSThread currentThread]); // 子线程
    }

@end

#####8、RunLoop的面试题
1.runloop与autoreleasepool的关系
>  
- 第一次创建:进入runloop的时候
- 最后一次释放:runloop退出的时候
- 其它创建和释放:当runloop即将休眠的时候会把之前的自动释放池释放,然后重新创建一个新的释放池

2.如何处理滑动UI过程中,广告轮播图停止轮询问题,使用runloop的哪种模式
> 这个问题其实看在"RunLoop中创建定时器"就可以了
- kCFRunLoopDefaultMode

3.runloop有几种模式,runloop接收几种输入源
> 
######  runloop有几种模式
- kCFRunLoopDefaultMode 默认模式,通常主线程在这个模式下运行
- kCFRunLoopCommonModes 占位符(表示)
- UITrackingRunLoopMode 界面跟踪Mode,用于追踪Scrollview的触摸滑动
- UIInitializationRunLoopMode:刚启动App时进入的第一个Mode,启动后不在使用。
- GSEventReceiveRunLoop:内部Mode,接收系事件。

> ######runloop接收几种输入源
 - Source0:非基于Port(处理事件)
 - Source1:基于Port (接收分发系统事件)

#####9、参考资料
[ CFRunLoopRef](http://opensource.apple.com/source/CF/CF-1151.16/)
[官方文档](https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/Multithreading/RunLoopManagement/RunLoopManagement.html)
[sunnyxxx线下视频](http://pan.baidu.com/s/1o8GRQ9W)
密码: jwnp
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容