文档章节

GCD的工作原理

JK_Jack
 JK_Jack
发布于 2015/10/19 17:26
字数 1484
阅读 53
收藏 0

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

                GCD                  

 
                         

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

?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
  - ( 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];
     });
}

?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
-( 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

?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
    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去得到队列,如下:

?

1
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

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

?

1
dispatch_queue_t mainQueut = dispatch_get_main_queue();

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

?

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

本文转载自:

共有 人打赏支持
JK_Jack
粉丝 0
博文 6
码字总数 2658
作品 0
广州
多线程&NSObject&NSThread&NSOperation&GCD

1、NSThread 每个NSThread对象对应一个线程,量级较轻(真正的多线程)以下两点是苹果专门开发的“并发”技术,使得程序员可以不再去关心线程的具体使用问题2、NSOperation/NSOperationQueu...

哥特复心
2013/12/30
0
1
iOS开发多线程篇---GCD的介绍和简单使用

1.GCD的简介: Grand Central Dispatch 简称(GCD)是苹果公司开发的技术,以优化的应用程序支持多核心处理器和其他的对称多处理系统的系统。这建立在任务并行执行的线程池模式的基础上的。而...

琳小兮
2015/03/05
0
1
iOS 多线程GCD

一.简介 GCD (Grand Central Dispatch)是Apple开发的多核编程的解决方法。 二.优点 1.GCD可用于多核的并行运算 2.GCD自动利用更多CPU内核(双核,四核) 3.GCD自动管理线程的生命周期(创建线...

龙飞凤舞de心
02/23
0
0
iOS面试题集合(BAT及各大中小型公司)

简介 本文主要内容为iOS面试题目,对各个面试题进行一些分类(持续更新),包含了BAT,及各大中小型公司的面试题集合,为大家在找工作的时候提供一点帮助,技术交流q群为150731459,大家互相交流学...

Oboe_b
2017/11/07
0
0
iOS多线程之 GCD (附实战实例)

GCD作为缩写意义有多种。它通常表示最大公约数(greatest common divisor,简写为gcd;或highest common factor,简写为hcf),此外它还是共产党的拼音缩写和游戏《鬼吹灯外传》的拼音缩写和“...

iOS雯Ping
03/06
0
0

没有更多内容

加载失败,请刷新页面

加载更多

JSCH会大量使用服务器内存吗?会

java实现一个需求用到了jsch,发现服务器内存会被占满。 写了个50进程的jsch-sftp测试连接 put一个文件 ExecutorService fixedThreadPool = Executors.newFixedThreadPool(50);for (int j =...

just-coding
15分钟前
1
0
聊聊redis的数据结构的应用

序 本文主要研究一下redis的数据结构的应用 string 最常用的就是incr操作,比如可以用来维护用户在某个抽奖活动的剩余抽奖次数 setnx方法可以用来实现分布式锁 hashmap 可以用来存储session...

go4it
15分钟前
1
0
《将博客搬至CSDN》

搬到csdn

我风依旧
18分钟前
1
0
源码编译安装最新版 Subversion 1.10.x

一、使用包管理器自动安装 官方网站提供了常见 Linux 发行版的安装命令,此处仅以 CentOS、Debian 及 Ubuntu 为例进行说明,其它系统详见官网:https://subversion.apache.org/packages.html...

whoru
22分钟前
0
0
liquibase

今天在看springboot源码时发现一个好东西。 LiquibaseServiceLocatorApplicationListener Liquibase是一个用于跟踪、管理和应用数据库变化的开源的数据库重构工具。它将所有数据库的变化(包...

jack_peng
26分钟前
0
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部