iOS开发线程和RunLoop

 

一般来讲,一个线程一次只能执行一个任务,执行完毕后线程就会退出,如果我们需要一个机制让线程能随时处理时间但并不退出,通常的代码逻辑是这样:

iOS开发线程和RunLoop_第1张图片

这就是 Event Loop框架。

runloop实际上就是一个管理其需要处理的事件和消息的对象,并提供了一个入口函数来执行上面Event loop的逻辑。线程执行了这个函数之后,就会一直处于这个函数内部“接受消息-》等待-》处理”的循环中,知道这个循环结束(例如传入quite消息),函数返回。

在OSX和iOS系统中,提供了两个这样的对象:NSRunLoop和CFRunLoopRef。

CFRunLoopRef是基于CoreFoundation框架内的,它提供了纯C函数的API,所有这些API都是线程安全的。

NSRunLoop是基于CFRunLoopRef的封装,提供面向对象的API,但是这些API不是线程安全的。

RunLoop和线程的关系:iOS开发中有两个线程对象-pthread_t和NSThread。pthread_tNSThread是一一对应的。比如你可以通过pthrea_main_thread_np()或者[NSThread currentThread]来获取当前线程。CFRunLoop是基于pthread来管理的。

苹果不允许直接创建RunLoop,但是提供了两个自动获取的函数:CFRunLoopGetMain()和CGRunLoopGetCurrent()。这两个函数的内部逻辑大概为:

iOS开发线程和RunLoop_第2张图片

线程和RunLoop之间是一一对应的,其关系是保存在一个全局的Dictionary里。线程刚创建时并没有RunLoop,如果你不主动获取,它一直都不会有。runloop的创建是发生在第一次获取的时候,销毁时在线程结束时。你只能在一个线程的内部获取其RunLoop(主线程除外)。

Event驱动  主体就是一个死循环,没事-休眠,有事-唤醒-执行  

runloop:用于结局

1、使程序一直运行并接受用户输入

2、决定程序在何时应该处理哪些Event

3、调用解耦(Message Queue)

4、节省cpu时间

 

setNeedsLayout/setNeedsDisplay主动跟跟新UI。

iOS程序入口在int main函数,main函数内部会rentun ,一旦return就会结束程序,所以内部有个mainrunloop一种在运行,不会造成return,很多crash都是因为runloop不执行了就return了造成程序无法执行下去。

iOS开发线程和RunLoop_第3张图片

如图,程序执行的时候不会执行nslog因为mainRunLoop在运行,不会ruturn一旦ruturn程序会退出。

  mainRunLoop在一直执行,随时唤醒处理事件,如果没有runloop就不能实现随时处理事件。

iOS开发线程和RunLoop_第4张图片

iOS开发线程和RunLoop_第5张图片

 

foundation中的NSRunLoop对象是通过Core Foundation框架的CFRunLoopRef封装而来,提供的api比较少。cfrunloopref用的比较多。 

iOS开发线程和RunLoop_第6张图片

子线程的runloop默认关闭,第一次获取的时候才创建,有时候用不到浪费资源。runloop有资源才不会销毁,没资源会自定销毁。

iOS开发线程和RunLoop_第7张图片

iOS开发线程和RunLoop_第8张图片

iOS开发线程和RunLoop_第9张图片

API里面NSTimer 是木有暂停继续的方法的,只有fire和invalidate,前者是开工的意思,后者是废掉的意思 

iOS开发线程和RunLoop_第10张图片

iOS开发线程和RunLoop_第11张图片

在这种场景:要获取事件(例如button按下之前需要处理,但是又不知道什么时候按下去),用runloop监听runloop的状态,唤醒了就说明按钮按下去了。

iOS开发线程和RunLoop_第12张图片

 

监听各个状态在某个状态中处理事件。

以下是个各种状态

iOS开发线程和RunLoop_第13张图片

iOS开发线程和RunLoop_第14张图片

performSelector:......也是source。

主线程的runloop默认开启,子线程的runloop默认不开启,所以timer在子线程中是不会执行的,需要手动开启runloop。手动开runloop需要runloop中添加资源,没有资源的runloop会自动关闭。

runloop应用:  runloop默认是不开启的,在线程中第一次获取的时候创建,run开启。

1、Timer

2、UIImageView滑动不加载,runloop DefaultMode才加载。

3、PerformSelect

4、常驻线程:让子线程常驻,需要让他的runloop运行起来,那么mode种需要有source,在里买呢添加一个NSPort就行。

  进程就是每个应用程序运行就会有一个进程,各应用间进程彼此独立,进程依靠线程执行任务,每个进程至少有一个线程,线程中任务的执行是串行的,如果要在1个线程中执行多个任务,那么只能一个一个地按顺序执行这些任务,也就是说,在同一时间内,1个线程只能执行1个任务。

  线程中执行任务只能串行,但是线程可以并发执行。 同一时间,CPU只能处理1条线程,只有1条线程在工作(执行),多线程并发(同时)执行,其实是CPU快速地在多条线程之间调度(切换),如果CPU调度线程的时间足够快,就造成了多线程并发执行的假象     

  思考:如果线程非常非常多,会发生什么情况?

  CPU会在N多线程之间调度,CPU会累死,消耗大量的CPU资源,每条线程被调度执行的频次会降低(线程的执行效率降低)。

 

多线程的优点

能适当提高程序的执行效率

能适当提高资源利用率(CPU、内存利用率)

 

多线程的缺点

开启线程需要占用一定的内存空间(默认情况下,主线程占用1M,子线程占用512KB),如果开启大量的线程,会占用大量的内存空间,降低程序的性能

线程越多,CPU在调度线程上的开销就越大

程序设计更加复杂:比如线程之间的通信、多线程的数据共享

5、自动释放池

 

转载于:https://www.cnblogs.com/yinqianios/p/6120133.html

你可能感兴趣的:(iOS开发线程和RunLoop)