iOS多线程GCD的简单使用

时间:2022-02-13 16:09:08

在iOS开发中,苹果提供了三种多线程技术,分别是:

(1)NSThread

(2)NSOperation

(3)GCD

简单介绍一下GCD的使用。

GCD全称 Grand Central Dispatch,可以称之为大*调度。实际上GCD是管理着一个线程池,如何创建线程,如何回收线程,以及分配多少个线程,这些都是GCD来控制的。在开发中,程序员是不用操作线程的相关事情,程序员只需要把应该做的操作放到相应的队列里面即可。

一:自定义队列

GCD中有多种队列,其中自定义的队列有两种:串行队列和并行队列

1:串行队列:队列中的任务只会顺序执行,且一次只能够执行一个任务。也就是说,执行完一个任务后,才会执行下一个任务

2:并行队列:可以一次执行多个任务。比如说并行队列中有10个任务,可以一次执行3个任务,这三个任务哪个先执行完了,再接着执行剩下的任务。

注意:无论是串行队列还是并行队列,他们都是FIFO(先进先出)的。也就是说,无论是哪种队列,任务进队列的时间越早,其执行的时间就越早(只不过某些情况下任务执行的结束时间是不确定的)。

GCD中有两种操作,分别是同步操作和异步操作

1:同步操作:不会新开线程

2:异步操作:会开启新的线程

两种操作和两种队列,组合为4种情况,实际上,在开发中,有些组合基本上是不会用到的。下面用程序描述一下四种组合。

组合一:串行队列+同步操作(不会新建线程,而且任务是一个一个的执行,因此实际上就是顺序执行),代码如下:

- (void)gcdDemo1
{
    //串行队列+同步操作
    dispatch_queue_t queue = dispatch_queue_create("gcddemo", DISPATCH_QUEUE_SERIAL);
    for(int i = 0; i < 10; ++i){
        dispatch_sync(queue, ^{
            NSLog(@"%@ %d",[NSThread currentThread],i);
        });
    }
}

执行结果:

iOS多线程GCD的简单使用

number = 1,说明是主线程,没有新开线程。

组合二:串行队列+异步操作(因为任务要一个一个的执行,但是因为是异步操作,所以会开启一个新的线程,所有的任务都在新的线程上执行),代码如下:

- (void)gcdDemo1
{
    dispatch_queue_t queue = dispatch_queue_create("gcddemo", DISPATCH_QUEUE_SERIAL);
    //串行队列+异步操作
    for (int i = 0; i < 10; ++i){
        dispatch_async(queue, ^{
            NSLog(@"%@ %d",[NSThread currentThread],i);
        });
    }
}

执行结果:

iOS多线程GCD的简单使用

number = 2,说明开启了一个新的子线程,但仍然是顺序执行。

组合三:并行队列+同步操作(因为同步操作不会开启新的线程,因此,即使并行队列可以一次开始多个任务,但实际上仍旧是每个任务都在主线程上执行,且按顺序执行)。代码如下:

- (void)gcdDemo2
{
    dispatch_queue_t queue = dispatch_queue_create("gcddemo", DISPATCH_QUEUE_CONCURRENT);
    //并行队列+同步任务
    for(int i = 0; i < 10; ++i){
        dispatch_sync(queue, ^{
            NSLog(@"%@ %d",[NSThread currentThread],i);
        });
    }
}

执行结果:

iOS多线程GCD的简单使用

没有开启新的线程,且按顺序执行。

组合四:并行队列+异步操作(并行队列会一次开始多个任务,且异步操作可以开启新的线程,因此同一时刻可能会同时执行多个任务,开启多个线程,且每个任务的结束时间是不确定的)。代码如下:

- (void)gcdDemo2
{
    dispatch_queue_t queue = dispatch_queue_create("gcddemo", DISPATCH_QUEUE_CONCURRENT);
    //并行队列+异步任务
    for(int i = 0; i < 10; ++i){
        dispatch_async(queue, ^{
            NSLog(@"%@ %d",[NSThread currentThread],i);
        });
    }
}

执行结果:

iOS多线程GCD的简单使用

可以看到,开启了多个线程,且任务不是按顺序执行完的。

二:全局队列

为了方便开发,苹果还提供了有全局队列,全局队列实际上是并行队列,因此,全局队列的执行结果和并行队列的执行结果是一致的。代码如下:

全局队列+同步任务:

- (void)gcdDemo3
{
    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    //全局队列+同步任务
    for (int i = 0; i < 10; ++i){
        //同步任务
        dispatch_sync(queue, ^{
            NSLog(@"%@ %d",[NSThread currentThread],i);
        });
    }
}

执行结果:

iOS多线程GCD的简单使用

全局队列+异步任务:

- (void)gcdDemo3
{
    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    //全局队列+异步任务
    for(int i = 0; i < 10; ++i){
        dispatch_async(queue, ^{
            NSLog(@"%@ %d",[NSThread currentThread],i);
        });
    }
}

执行结果:

iOS多线程GCD的简单使用

三:主队列

苹果还提供了一种队列是主队列,主队列是串行队列,但是和串行队列又有差异。主队列上的任务都应该在主线程上顺序执行,没有异步的概念。也就是说,即使是异步任务在主队列上执行,也不会开启新的线程。

主队列+异步任务:

- (void)gcdDemo4
{
    dispatch_queue_t queue = dispatch_get_main_queue();
    //主队列+异步任务
    for(int i = 0; i < 10; ++i){
        dispatch_async(queue,^{
            NSLog(@"%@ %d",[NSThread currentThread],i);
        });
    }
}

执行结果:

iOS多线程GCD的简单使用

可以看到,没有开启新的线程,且是顺序执行。

主队列+同步任务(会阻塞线程),代码如下:

- (void)gcdDemo4
{
    dispatch_queue_t queue = dispatch_get_main_queue();
    //主队列+同步任务,会阻塞
    for(int i = 0; i < 10; ++i){
        dispatch_sync(queue, ^{
            NSLog(@"%@ %d",[NSThread currentThread],i);
        });
    }
}

阻塞原因:

主队列中本身是有一个任务A的(主任务),且该任务A还没有执行完。在执行任务A的过程中,又插入了新的同步任务B。我们知道,串行队列中,必须先执行完一个任务后,才能继续执行另一个任务。此时的情况时:

若想继续执行任务A,需要先把任务B执行完,若想继续执行任务B,需要先把任务A执行完,因此造成了阻塞。

在开发中,应该避免这种阻塞的情况。