iOS 多线程学习笔记 —— GCD

本文复制、参考自文章:iOS多线程编程之Grand Central Dispatch(GCD)介绍和使用 ,主要为了增强我的对知识的理解和记忆,不作他用。原做者声明:html

著做权声明:本文由http://blog.csdn.net/totogo2010/原创,欢迎转载分享。请尊重做者劳动,转载时保留该声明和做者博客连接,谢谢!ios

这里对原做者的辛勤工做表示感谢!数据库

1. 简介

GCD (Grand Central Dispatch) 是创建任务并行执行的线程池模式的基础上的,以优化支持多核、多处理器系统。编程

GCD的工做原理是:让程序平行排队的特定任务,根据可用的处理资源,安排他们在任何可用的核心上执行任务。一个任务能够是一个函数(function)或者是一个block。网络

GCD的底层依然是用线程实现,不过没必要关心实现细节。多线程

GCD的FIFO队列称为 调度队列(dispatch queue), 它能够保证先来的先执行。dispatch 的、queue 分为如下3种:并发

 (1) Serial (连续调度队列)

 又称为 私有调度队列(pirvate dispatch queue), 同时只执行一个任务。Serial queue一般用于同步访问特定的资源或数据。当建立多个Serial queue时,虽然它们各自队列内是单独执行的,但队列间是并发执行的。app

 (2) Concurrent (并发调度队列)

 又称为 全局调度队列(global dispatch queue), 能够并发地执行队列中的多个任务,可是执行完成的顺序是随机的。异步

 (3) Main dispatch queue(主调度队列)

 它是全局可用的serial queue,是在应用程序主线程上执行任务的。async

2. 调度队列的使用

2.1 经常使用的方法 dispatch_async

  为了不界面在处理耗时的操做时卡死,好比读取网络数据、IO、数据库读写等,咱们会在另外一个线程中去处理这些操做,而后通知主线程更新界面。

  用GCD实现这个流程比前面介绍的NSThread\NSOperation的方法都要简单,代码以下:

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        // 耗时的操做

  dispatch_async(dispatch_get_main_queue(), ^{
       // 更新界面
  });
});

针对以前图片加载的示例:

- (void)viewDidLoad
{
    [super viewDidLoad];
    
    // Do any additional setup after loading the view, typically from a nib.
//    NSThread* thread = [[NSThread alloc] initWithTarget:self selector:@selector(downloadImage:) object:kURL];
//    [thread start];
    
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        NSURL* url = [NSURL URLWithString:kURL];
        NSData* data = [[NSData alloc] initWithContentsOfURL:url];
        UIImage* image = [[UIImage alloc] initWithData:data];
        if (data != nil)
        {
            dispatch_async(dispatch_get_main_queue(), ^{
                self.imageView.image = image;
            });
        }
    });
    
    
}

GCD会自动根据任务在多核处理器上分配资源,优化程序。

系统给每一个应用程序提供了三个并发调度队列(concurrent dispatch queues),这3个队列是全局的,它们只有优先级的不一样。由于是全局的,咱们不须要去建立,只须要经过使用函数  dispatch_get_global_queue 去获得队列。以下:

dispatch_queue_t globalQ = dispatch_get_global_queue(DISPATCH_QUEU_PRIORITY_DEFAULT, 0);

系统还提供了一个主调度队列 main_queue

dispatch_queue_t mainQueue = dispatch_get_main_queue();

这些队列都是全局队列,,不用retain或release。

2.2 dispatch_group_async的使用

 dispatch_group_async能够实现监听一组任务是否完成,完成后再通知执行其余的操做。以下示例:

    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    dispatch_group_t group = dispatch_group_create();
    
    dispatch_group_async(group, queue, ^{
        [NSThread sleepForTimeInterval:1];
        NSLog(@"group1");
    });
    
    dispatch_group_async(group, queue, ^{
        [NSThread sleepForTimeInterval:1];
        NSLog(@"group2");
    });
    
    dispatch_group_async(group, queue, ^{
        [NSThread sleepForTimeInterval:1];
        NSLog(@"group3");
    });
    
    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
        NSLog(@"updateUI");
    });

前3个dispatch_group_async执行是异步的,顺序不定。

2.3 dispatch_barrier_async的使用

dispatch_barrier_async 是在前面的任务执行结束后它才执行,并且它后面的任务等它执行完成以后才会执行。示例代码:

    dispatch_queue_t queue = dispatch_queue_create("gcdtest.rongfzh.yc", DISPATCH_QUEUE_CONCURRENT);
    
    dispatch_async(queue, ^{
        [NSThread sleepForTimeInterval:1];
        NSLog(@"dispatch_async1");
    });
    
    dispatch_async(queue, ^{
        [NSThread sleepForTimeInterval:1];
        NSLog(@"dispatch_async2");
    });
    
    dispatch_async(queue, ^{
        [NSThread sleepForTimeInterval:1];
        NSLog(@"dispatch_async3");
    });
    
    dispatch_barrier_async(queue, ^{
        NSLog(@"dispatch_barrier_async");
        [NSThread sleepForTimeInterval:4];
    });
    
    dispatch_async(queue, ^{
        [NSThread sleepForTimeInterval:1];
        NSLog(@"dispatch_async4");
    });

一样的,前3个dispatch_async执行是异步的,顺序不定。

2.4 dispatch_apply

执行某个代码片断N次。

    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    dispatch_apply(5, queue, ^(size_t index) {
        NSLog(@"dispatch_apply: index: %d", index);
    });

 

GCD还有不少其余用法,能够参考官方文档

参考的文档还有:http://en.wikipedia.org/wiki/Grand_Central_Dispatch