美文网首页
理理iOS的多线程解决方案 - GCD

理理iOS的多线程解决方案 - GCD

作者: 一只呱呱 | 来源:发表于2018-02-04 15:21 被阅读0次

    周末梳理了一下iOS几种多线程的方案,总结下平时GCD的主要用法和场景。

    一、iOS常见的多线程方案

    (1) Pthreads: 基于C语言的框架,在多种操作系统上都有使用,要手动地去管理生命周期,所以在iOS平常的开发中并不常用。

    (2) NSThread: Apple封装后的线程对象,常用的几种简单的获取线程,启动取消线程等方法。开发中只有一些简单的场景例如[NSThread currentThread], performSelectorOnMainThread:才会使用,因为它在解决问题时还不够智能,要手动地去管理状态。

    (3) GCD: Apple为并行计算提出的解决方案,优点在于它会自动管理线程的生命周期。同时它使用了Block来让使用更加方便。

    (4) NSOperation & NSOperationQueue: Apple对GCD进行了更多一层的封装,提供了更多的接口来进行任务的管理。

    二、GCD

    (1) GCD中两个重要的基本概念:队列和任务

    * 队列:

    一种遵循FIFO来存放任务的队列,GCD中有两种类型:串行队列(Serial)和并行队列(Concurrent)

    串行队列:对应一个线程,让任务一个接一个地执行。系统默认提供main_queue,并规定UI只能在主线程中操作,避免产生混乱

    并行队列:可能对应多个线程,可以让多个任务同时执行。系统默认提供global_queue

    我们自己可以创建串行和并行队列

    * 任务:

    就是需要执行的一段代码,GCD是放在Block里的。执行任务有两种方式同步执行(async)和异步执行(async)

    如果是同步执行,当前任务会阻塞当前线程并等待Block中的任务执行完成才继续下一个任务

    如果是异步执行,当前任务不会阻塞等待,会直接往下执行。

    "放到并行队列的异步(async)任务,GCD 也会 FIFO的取出来,但不同的是,它取出来一个就会放到别的线程,然后再取出来一个又放到另一个的线程。"

    例1:阻塞主线程

    分析:这里在主线程发起了一个主线程中的同步任务,当要开始执行的时候,dispatch_sync阻塞了主线程,并尝试把block中的任务放到主线程中执行,但这时候主线程已经被阻塞,所以任务无法完成,造成死锁。

    同理:

    例2:asyn和async的任务决定了什么?

    从以上的结果看来,queue只是管理任务的一个方式,无法决定执行的某一个线程,但是能决定任务顺序以及是否开辟新的线程。

    (2) GCD常用的几个并行队列中线程同步的方法:Group、Barrier、Semaphore

    dispatch_barrier_async/sync

    在GCD中起到一个栅栏的作用,同Concurrent Dispatch Queue队列一起使用。它等待所有位于barrier函数之前的操作执行完毕后执行,并且在barrier函数执行之后,barrier函数之后的操作才会得到执行。

    barrier的async和sync的区别只是,是否在执行结束才插入别的任务。但是都会保证同时这个队列只有这个任务在执行

    假设有一块资源需要同步读异步写,则可以使用barrier函数来保证写操作之前的读操作全部完成后,才进行写。不会有两次读取的值不一样的问题。

    dispatch_group

    可以把任务分组,打包成为一个大的任务,dispatch_group_notify来保证前面的异步/同步任务都结束后才回到某个线程。但是无法保证其中异步任务执行的顺序

    dispatch_semaphore

    有时候线程过多,需要控制Concurrent Queue中运行的线程数量,就可以使用semaphore。而以下的这些在Serial Queue中是不生效的,因为Serial Queue中始终一次只能执行一个任务

    1. dispatch_semaphore_create 创建一个semaphore

    2. dispatch_semaphore_signal 发送一个信号

    3. dispatch_semaphore_wait 等待信号

    得出的下面结果看到,semaphore可以控制同时执行的线程数量,换言之也可以用来保证线程的同步。

    假设把信号量改成2,则可以看到有两个任务可以同时进行

    用信号量可以实现两个线程之间的依赖关系

    (3)dispatch_once 和 dispatch_after,dispatch_suspend

    dispatch_once 

    常见的单例创建方法

    dispatch_after

    dispatch_after是来延迟执行的GCD方法,因为在主线程中我们不能用sleep来延迟方法的调用,所以用dispatch_after是最合适的,dispatch_after的真正含义是在6秒后把任务添加进队列中,并不是表示在6秒后执行。

    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

     NSLog(@"after 2 seconds %@",[NSThread currentThread]);

     });

    dispatch_suspend

    suspend不会立即暂停正在运行的block,而是在当前block执行完成后,暂停后续的block执行。

    (4)线程之间通信

    (5)如何停止一个GCD中正在运行的线程

    用一个全局变量或是property来控制

    @property(nonatomic, assign)BOOL shouldCancel;
    @property(nonatomic, strong)dispatch_queue_t conQueue;

    - (void)viewDidAppear:(BOOL)animated  {  

        [super viewDidAppear:animated];  

        self.conQueue = dispatch_get_global_queue(0, 0);  

        dispatch_async(self.conQueue, ^{  

            while (!self.shouldCancel) {  

                    sleep(1);  

            };  

        });  

    }  

    - (void)closeThread  {  

    if (self.conQueue) {  

            self.shouldCancel = YES;  

            dispatch_suspend(q);  

            dispatch_release(q);  

            self.conQueue = nil;  

        }  

    }  

    Reference

    https://www.jianshu.com/p/0b0d9b1f1f19

    https://www.jianshu.com/p/2d57c72016c6

    https://www.jianshu.com/p/cfcc0c302621

    http://blog.csdn.net/linfengwenyou/article/details/48948355

    相关文章

      网友评论

          本文标题:理理iOS的多线程解决方案 - GCD

          本文链接:https://www.haomeiwen.com/subject/znsmzxtx.html