iOS开发多线程---GCD的介绍和简单使用

jopen 9年前

GCD的工作原理是:让程序平行排队的特定任务,根据可用的处理资源,安排他们在任何可用的处理器核心上执行任务。一个任务可以是一个函数(function)或者是一个block。 GCD的底层依然是用线程实现,不过这样可以让程序员不用关注实现的细节。GCD中的FIFO队列称为dispatch queue,它可以保证先进来的任务先得到执行。用gcd实现网络数据请求,比前面介绍的NSThread方法要简单的多。

1.GCD的简介:

 Grand Central Dispatch 简称(GCD)是苹果公司开发的技术,以优化的应用程序支持多核心处理器和其他的对称多处理系统的系统。这建立在任务并行执行的线程池模式的基础上的。而且是纯C语言,提供了非常多强大的函数。

2.GCD的工作原理是:

让程序平行排队的特定任务,根据可用的处理资源,安排他们在任何可用的处理器核心上执行任务。一个任务可以是一个函数(function)或者是一个block。 GCD的底层依然是用线程实现,不过这样可以让程序员不用关注实现的细节。

3.GCD的优势:

GCD是苹果公司为多核的并行运算提出的解决方案;

GCD会自动利用更多的CPU内核(比如双核、四核);

GCD会自动管理线程的生命周期(创建线程、调度任务、销毁线程);

程序员只需要告诉GCD想要执行什么任务,不需要编写任何线程管理代码;

温馨提示:

GCD存在于libdispatch.dylib这个库中,这个调度库包含了GCD的所有的东西,但任何IOS程序,默认就加载了这个库,在程序运行的过程中会动态的加载这个库,不需要我们手动导入。

4.GCD的使用步骤和核心概念:

核心概率有2个:1)队列:用来存放任务  2)任务:执行什么操作

使用步骤有2步:1)定制任务 2)确定想做的事情

5.dispatch queue(队列)

GCD中的FIFO队列称为dispatch queue,它可以保证先进来的任务先得到执行,dispatch queue分为下面三种:

1)Serial dispatch queues(串行队列),同时只执行一个任务(一个任务执行完毕以后,再执行下一个任务)。Serial queue通常用于同步访问特定的资源或数据。当你创建多个Serial queue时,虽然它们各自是同步执行的,但Serial queue与Serial queue之间是并发执行的。

2)Concurren dispatch queue(并发队列),可以并发地执行多个任务(可以让多个任务同时执行,自动开启多个线程同时执行任务,并发功能只有在异步函数(dispatch_async)下有效),但是执行完成的顺序是随机的。

3)Main dispatch queue(主队列),它是全局可用的serial queue(直接在主线程中串行执行任务),它是在应用程序主线程上执行任务的。

6.GCD执行任务的两种方式:

//将参数block(任务)提交给参数queue(队列)进行执行,参数说明:queue:队列   block:任务

(1)用同步的方式执行任务 dispatch_sync(dispatch_queue_t queue, dispatch_block_t block);

(2)用异步的方式执行任务 dispatch_async(dispatch_queue_t queue, dispatch_block_t block);

7.为了避免界面在处理耗时的操作时卡死,比如读取网络数据,数据库读写等,我们会在另外一个线程中处理这些操作,然后通知主线程更新界面。创建一个全局并发队列,使用dispatch_async执行下载图片任务,然后再回到主线程中展示图片。

第一种方式---GCD结合NSThread

 - (void)viewDidLoad  {      [super viewDidLoad];         imagev = [[UIImageView alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];      [self.view addSubview:imagev];         //说明:全局并发队列的优先级  #define DISPATCH_QUEUE_PRIORITY_HIGH 2 // 高  #define DISPATCH_QUEUE_PRIORITY_DEFAULT 0 // 默认(中)  #define DISPATCH_QUEUE_PRIORITY_LOW (-2) // 低  #define DISPATCH_QUEUE_PRIORITY_BACKGROUND INT16_MIN // 后台            //第一个参数全局并发队列的优先级,第二个参数暂时无用,用0即可      dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);      //使用异步的方式dispatch_async执行任务,第一个参数获得全局的并发队列,第二个参数block(队列将要执行的任务)      dispatch_async(queue, ^{          //调用下载图片方法          [self downImage];      });  }
-(void)downImage  {      //从网络中下载图片      NSURL *url = [NSURL URLWithString:@"http://i8.topit.me/8/c1/31/1142319854bdc31c18o.jpg"];      //将图片转换为二进制数据      NSData *imgData = [NSData dataWithContentsOfURL:url];      //数据转换成图片      UIImage *img = [UIImage imageWithData:imgData];            //回到主线程设置图片      [self performSelectorOnMainThread:@selector(senderImage:) withObject:img waitUntilDone:NO];  }    -(void)senderImage:(UIImage *)image  {      imagev.image = image;  }

第二种方式---GCD

   dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{                    //从网络中下载图片          NSURL *url = [NSURL URLWithString:@"http://i8.topit.me/8/c1/31/1142319854bdc31c18o.jpg"];          //将图片转换为二进制数据          NSData *imgData = [NSData dataWithContentsOfURL:url];          //数据转换成图片          UIImage *img = [UIImage imageWithData:imgData];                    dispatch_async(dispatch_get_main_queue(), ^{              //回到主线程中设置图片显示              imagev.image = img;          });      });

总结:系统给每一个应用程序提供了三个concurrent dispatch queues。这三个并发调度队列是全局的,它们只有优先级的不同。因为是全局的,我们不需要去创建。我们只需要通过使用函数dispath_get_global_queue去得到队列,如下:

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

上面的例子中我们也用到了,系统默认就有的一个串行队列main_queue

dispatch_queue_t mainQueut = dispatch_get_main_queue();

他们的代码框架结构如下:

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{        //这里就做一些比较耗时的操作,如请求数据。。        dispatch_async(dispatch_get_main_queue(), ^{            //这里就是回到主线程去更新界面        });    });