阅读 557

中高级iOS必备知识点之 RunLoop(一)

RunLoop学习起来是很抽象,也不好理解,所以一定多看几次,多学学才能学好!这也是中高级iOS必须掌握的知识点,面试中经常遇到.

什么是 RunLoop?

Run 表示运行,Loop 表示循环。结合在一起就是运行循环的意思。RunLoop就是在程序运行过程中循环做一些事情.

RunLoop的应用范畴有哪些?

定时器(Timer)、PerformSelector

GCD Async Main Queue

事件响应、手势识别、界面刷新

网络请求

AutoreieasePool

上面这些底层都是RunLoop在支撑,说白了,如果没有RunLoop支撑,上面的这些都无法实现.

如果没有RunLoop会发生什么呢?像我们的命令行项目,创建出来默认就是没有RunLoop,请看下图

1.png

因为没有RunLoop,程序执行到第13行的时候,就会自动退出.

而我们iOS项目的main函数里面都有UIApplicationMain(argc, argv,nil, appDelegateClassName);这个代码,这里就是创建了一个主线程的RunLoop,所以我们程序不会退出,一直在运行中.我们可以大致写一下main函数里面的伪代码如下:

2.png

retVal这个等于0,当没有事件处理的时候,RunLoop就会sleep就是类似睡觉,一旦有事件需要处理,比如点击、刷新事件等process_message就会去处理这个事件,处理完了继续休息,retVal=0,程序就会一直执行,不会退出,这就是RunLoop作用.

RunLoop的基本作用

1.保持程序的持续运行

2.处理App中的各种事件(比如触摸事件、定时器事件等)

3.节省了CPU资源,提高程序性能:该做事时做事,该休息时休息


获取RunLoop对象

iOS中有2套API来访问和使用RunLoop :

Foundation : NSRunLoop (OC语言里面的)

Core Foundation : CFRunLoopRef (C语言里面的)

NSRunLoop和CFRunLoopRef都代表着RunLoop对象

NSRunLoop是基于CFRunLoopRef的一层OC包装

CFRunLoopRef是开源的.(CFRunLoopRef参考链接)

其实我们很多都是由OC包装出来的,请看下面:

3.png

获取当前的RunLoop

获取当前RunLoop和主线程RunLoop

4.png

获取RunLoop

这里注意,地址不一样,因为NSRunLoop是对CFRunLoopDef做了一层包装,你可以用OC的NSLog("%@",[NSRunLoop MainRunLoop])获取对比一下,它的地址就是C语言获取的地址.主线程只有一个RunLoop.

RunLoop与线程

每条线程都有唯一的一个与之对应的RunLoop对象(一一对应)

RunLoop保存在一个全局的Dictionary里,线程作为key,RunLoop作为value

线程刚创建的时候并没有RunLoop对象,RunLoop会在第一次获取它时创建

RunLoop会在线程结束时销毁

主线程的RunLoop已经自动创建,子线程默认没有开启RunLoop.

5.png

源码窥探看一下:CFRunLoopGetCurrent

由于源码不能像objc直接打开,我们把它拉到项目中查看.

6.png

7.png

从字典也能看出来是一对一的关系.而且确实是第一次获取的时候是空的,然后再去创建这个RunLoop.

那我们就继续来了解RunLoop内部的数据结构,到底是怎么工作的.

RunLoop相关的类

Core Foundation中关于RunLoop的5个类

1.CFRunLoopRef

2.CFRunLoopModeRef

3.CFRunLoopSourceRef

4.CFRunLoopTimerRef

5.CFRunLoopObserverRef

再看下CFRunLoopRef的底层源码:

8.png

就是上面这个结构体,我们用到的可能就是红色这些.pthread是线程,每个runloop都会保存这个东西.最后面那个_modes,这个是个集合来着,CFMutableSetRef我们能想到我们自己用的set也是一个集合来着,比如NSMutableSet也是一个集合,所以这个_modes里面是存着一堆的mode.

这个mode就是CFRunLoopModeRef类型,所以里面存储一堆的CFRunLoopModeRef类型的mode.

而_currentMode也是CFRunLoopModeRef这个类型,所以我们很容易得出一个结论:

一个RunLoop对象里面有一堆的mode,也就是存在_modes里面,里面只有一个是_currentMode.

我们再窥探一下源码,看下mode里面存储的是什么?

9.png

所以我们来个总结的图:

10.png

RunLoop有很多种模式,对应的_currentMode只有一种.

CFRunLoopModeRef

1.CFRunLoopModeRef它是代表RunLoop的运行模式

2.一个RunLoop包含若干个Mode,每个Mode又包含若干个Source0/Source1/Timer/Observer

3.RunLoop启动时只能选择其中一个Mode,作为currentMode

4.如果需要切换Mode,只能退出当前RunLoop,再重新选择一个Mode进入

5.不同组的Source0/Source1/Timer/Observer能分割开来,互不影响

6.如果Mode里面没有任何Source0/Source1/Timer/Observer,RunLoop会立马退出

如果只能在一种模式下运行,对性能什么的都有很大好处,比如我在滑动模式下,不考虑不滑动的模式,所以就不会卡顿,顺畅很多.还有注意的就是,它切换mode是在循环里面切换的,所以不会导致程序退出.

常见的mode有2种,其他情况很少见,所以掌握这两个一般都是没问题了

1.KCFRunLoopDefaultMode (NSDefaultRunLoopMode):App的默认Mode,通常是主线程是在这个Mode下运行

2.UITrackingRunLoopMode : 界面跟踪Mode,用于ScrollView追踪触摸滑动,保证界面滑动时不受其他Mode影响

RunLoop到底做哪些事?

RunLoop在不停执行的时候到底具体做了哪些事?其实是RunLoop在不停循环的时候,就是处理每个mode下的Source0、Source1、Timer、Observer这里面的事件,那我们就来看看这里面具体对应的到底是什么事件.

Source0

触摸事件、performSelector:onThread:

比如我们的touchbegin这个我们看下下面的代码:

11.png

Source1

基于Port的线程间的通信,系统事件的捕捉.

(两个线程之间相互传递消息的处理,系统事件捕捉,其实也包括触摸事件,只是把事件捕捉到以后传递给Source0).

Timer

NSTimer定时器,performSelector:withObject:afterDelay(这个方法的底层实现也就是NSTimer来实现的)

Observers

用于监听RunLoop的状态,UI的刷新(BeforeWaiting),Autorelease pool(BeforeWaiting)

(在RunLoop休眠之前都会去执行UI的刷新啊、Autorelease pool的释放等)

以上这些东西,完全就是我们平时开发中经常写的代码,比如设置背景色,设置frame等等.

文章分类
iOS
文章标签