• 什么是GCD

  Grand Central Dispatch 是Apple开发的一种多核编程技术。主要用于优化应用程序以支持多核处理器以及其他多对称处理系统TA会自动管理线程的生命周期(创建线程、调度任务、销毁线程),完全不需要我们管理,我们只需要告诉干什么就行。同时它使用的也是 c语言,不过由于使用了 Block(Swift里叫做闭包),使得使用起来更加方便,而且灵活。所以基本上大家都使用 GCD 这套方案

  • 什么是任务和队列

  在GCD中有两个非常重要的概念,就是任务和队列

  任务:任务就是一段功能代码,在GCD中一般就是一段block或者函数(block用的比较多),任务有两种执行方式,同步和异步。

  同步和异步的区别就在于是否会阻塞当前线程:

  如果是 同步(sync) 操作,它会阻塞当前线程并等待 Block 中的任务执行完毕,然后当前线程才会继续往下运行。

  如果是 异步(async)操作,当前线程会直接往下执行,它不会阻塞当前线程。

  同步函数无法开启子线程而异步函数有开启子线程的功能。

  队列:用于存放任务,队列一般分为两种,串行队列(SerialQueue)和并行队列(Concurrent),都遵守FIFO。

  队列中的任务根据同步和异步的不同有不同的执行方式:

  

  同步并行,因为同步没有开启子线程的功能,所以并行队列也就失去了并行的效果,一般使用最多的方式就是异步并行。

  

  主队列:这是一个比较特殊的队列,主队列是主线程中的队列,所以不管是异步函数还是同步函数,都不会生成新的分支线程,也就是说异步函数在主队列中丧失了开启线程的能力。TA用于刷新 UI,任何需要刷新 UI 的工作都要在主队列执行,所以一般耗时的任务都要放到别的线程执行。

  •  GCD的使用

  1、异步并发

//1、异步并发(最常用) 开启多条子线程,任务是并发执行的
- (void)asynGlobal {
//获取全局的并发队列
//第一个参数代表队列优先级,如果优先级高,那么这个队列调度的次数就会多一点。通常情况下,这个优先级给默认的优先级
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, );
//一般情况下并行队列使用这个全局的并发队列就行了,不想使用全局队列也可以创建,因为是c函数,所以第一个参数的字符串不要顺手就在前面加个@符号
// dispatch_queue_t queue = dispatch_queue_create("6666", DISPATCH_QUEUE_CONCURRENT);
//将任务添加到并发队列中异步执行
dispatch_async(queue, ^{
NSLog(@"%@-----你484傻-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_async(queue, ^{
NSLog(@"%@------你没吃药吧----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_async(queue, ^{
NSLog(@"%@-----能不能去吃药-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_async(queue, ^{
NSLog(@"%@-----滚去吃药-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_async(queue, ^{
NSLog(@"%@-----扑街-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_async(queue, ^{
NSLog(@"%@-----再次扑街-----%d",[NSThread currentThread], [NSThread isMainThread]);
}); //如果是MRC的环境,队列使用完毕之后,需要释放队列
// dispatch_release(queue);
}   //执行效果 如果加入队列的任务太多,为了不消耗过多性能,系统会控制开启的子线程的条数,等有任务执行完了再让剩下的任务继续并行,所以才会有第一条和最后一条是同一个线程的结果。由于是异步函数开启了子线程不是在主线程中,所以[NSThread isMainThread]的值为0
  

  2、异步串行

//异步串行(有时会用) 开启一条子线程,任务在开启的线程中挨个执行
- (void)asynSerial {
//创建一个串行队列
dispatch_queue_t queue = dispatch_queue_create("bada.queue", DISPATCH_QUEUE_SERIAL); //将任务添加到串行队列中异步执行
dispatch_async(queue, ^{
NSLog(@"%@-----你484傻-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_async(queue, ^{
NSLog(@"%@------你没吃药吧----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_async(queue, ^{
NSLog(@"%@-----能不能去吃药-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_async(queue, ^{
NSLog(@"%@-----滚去吃药-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_async(queue, ^{
NSLog(@"%@-----扑街-----%d",[NSThread currentThread], [NSThread isMainThread]);
}); }
  //从结果就能看出来是在同一条子线程执行的
  

  

  3、同步并发

  

//同步 并发(基本不使用) 如果是同步,在不开启子线程的情况下,并发的效果已经消失了
- (void)synGlobal { //获取全局的并发队列
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, );
//将任务添加到并发队列中同步执行
dispatch_sync(queue, ^{
NSLog(@"%@-----你484傻-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_sync(queue, ^{
NSLog(@"%@-----你没吃药吧-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_sync(queue, ^{
NSLog(@"%@-----能不能吃药-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_sync(queue, ^{
NSLog(@"%@-----滚去吃药-----%d",[NSThread currentThread], [NSThread isMainThread]);
});
dispatch_sync(queue, ^{
NSLog(@"%@-----扑街----%d",[NSThread currentThread], [NSThread isMainThread]);
}); }
  //从打印结果就能看出来,由于是同步,并没有开启子线程,都是在主线程执行的,并发的效果也已经没了
  
  •  先来个例子

  当你创建了一个串行队列,先往队列中添加了一个异步任务,然后在异步任务执行的时候又加入了一个同步任务会是什么情况呢?

 - (void)test {
dispatch_queue_t queue = dispatch_queue_create("自己创建的串行队列", DISPATCH_QUEUE_SERIAL);
NSLog(@"11111111 - %@",[NSThread currentThread]);
dispatch_async(queue, ^{
NSLog(@"3333333333 - %@",[NSThread currentThread]);
dispatch_sync(queue, ^{
NSLog(@"---------- %@",[NSThread currentThread]); });
NSLog(@"----------- %@",[NSThread currentThread]); });
NSLog(@"22222222 - %@",[NSThread currentThread]);
}

  如果你还不太熟悉GCD,对于上面的结果也许会懵圈

  咱们来分析一下上面这段代码的执行顺序,第二行第三行会依次执行这没有问题,碰到第四行有个异步任务,因为是异步会开辟出一个子线程,并不会阻塞当前的线程任务,所以异步block内的内容会放到子线程中执行,跟主线程的任务互相不会干扰。子线程中先执行了第五行的打印,然后碰到了一个同步任务,但是同步会阻塞当前线程的任务往下执行,要先执行了同步block中的内容才能继续往下执行。然而当前只有一条线程,同步任务排在异步任务后面,同步任务要等排在前面一个的异步任务执行完成之后才会执行,这样就出现了两个任务互相等待的情况,这样谁都无法往下执行下去,就出现了卡死的现象,这叫做死锁。

  •  再来个例子

  

   这次是在一个并行队列中先执行了一个异步任务,同样也加入了一个同步任务

 - (void)test1 {
dispatch_queue_t queue = dispatch_queue_create("自己创建的并行队列", DISPATCH_QUEUE_CONCURRENT);
NSLog(@"1111111111111 - %@",[NSThread currentThread]);
dispatch_async(queue, ^{
NSLog(@"333333333333 - %@",[NSThread currentThread]);
dispatch_sync(queue, ^{
NSLog(@"44444444444444444 - %@",[NSThread currentThread]);
for (int i = ; i < ; i ++) {
NSLog(@"%d, 55555555555555 %@",i, [NSThread currentThread]);
}
dispatch_sync(queue, ^{
NSLog(@"");
});
});
NSLog(@"77777777777777 - %@",[NSThread currentThread]); });
NSLog(@"2222222222222 - %@",[NSThread currentThread]);
}

  你会发现,这次并没有发生上面那种死锁的情况,如果你想不明白了,那么咱们再来分析分析

  首先,还是从主线程开始执行这依然不会有什么问题,第四行的时候遇到了一个异步任务,此时就会开启一条子线程,主线程的任务会继续下去,异步block的代码会在刚开辟的子线程中执行,主线程和子线程的任务是并行的,所以你们执行这段代码的时候出来的打印结果的顺序可能跟我上面的打印结果会有不同。在子线程中,打印完第五行的结果时,碰到了同步任务,这跟第一个例子就有所不同了,由于是在并行队列中,异步任务会挂起,给同步任务让道让TA先执行,执行同步任务的时候又遇到了一个同步任务,当前正在执行的同步任务也会挂起,所以不会发生锁死的现象

  暂结。。。。如果哪里有错误请指教。。。。

多线程之GCD的更多相关文章

  1. iOS多线程之GCD小记

    iOS多线程之GCD小记 iOS多线程方案简介 从各种资料中了解到,iOS中目前有4套多线程的方案,分别是下列4中: 1.Pthreads 这是一套可以在很多操作系统上通用的多线程API,是基于C语言 ...

  2. iOS 多线程之GCD的使用

    在iOS开发中,遇到耗时操作,我们经常用到多线程技术.Grand Central Dispatch (GCD)是Apple开发的一个多核编程的解决方法,只需定义想要执行的任务,然后添加到适当的调度队列 ...

  3. iOS-多线程之GCD(原创)

    前言 GCD 全称 Grand Central DisPath NSOperation便是基于GCD的封装 基础知识 1.GCD的优势 (1)为多核的并行运算提出了解决方案 (2)GCD会自动利用更多 ...

  4. iOS多线程之GCD详解

    GCD(Grand Central Dispatch)是基于C语言开发的一套多线程开发机制.也是目前苹果官方推荐的多线程开发方法.iOS三种多线程开发中GCD是抽象层次最高的.当然用起来也是最简单的. ...

  5. IOS中的多线程之GCD

    在ios中,使用多线程有三种方式,分别是:NSThread.NSOperation和NSOperationQueue.GCD,在本节,主要讲解一下CDD的使用. GCD(Grand Central D ...

  6. iOS多线程之GCD学习笔记

    什么是GCD 1.全称是Grand Central Dispatch,可译为“牛逼的中枢调度器” 2.纯C语言,提供了非常多强大的函数 GCD的优势 GCD是苹果公司为多核的并行运算提出的解决方案 G ...

  7. IOS 多线程之GCD

    参考:http://www.cnblogs.com/wendingding/p/3806821.html <<Objective-C基础教程>> 第二版 一 简介 GCD 全称 ...

  8. (五十五)iOS多线程之GCD

    GCD的全称为Grand Central Dispatch,翻译为大中央调度,是Apple开发的一个多线程编程解决方法. 进程和线程的概念: 正在进行中的程序被称为进程,负责程序运行的内存分配,每一个 ...

  9. iOS开发多线程之GCD

    Grand Central Dispatch(GCD)是异步执行任务的技术之一.一般将应用程序中记述的线程管理用的代码在系统级中实现.开发者只需要定义想执行的任务并追加到适当的Dispatch Que ...

随机推荐

  1. Centos 安装jdk1.8

    我是根据右边链接进行安装的 ,但是第一步不同噢.http://www.cnblogs.com/spiders/archive/2016/09/06/5845727.html 1.下载rpm安装文件. ...

  2. 关于ActionBar

    添加ActionBar: Android 3.0(API 11)(不含API11)以下的版本中,如果需要活动有ActionBar,需要让活动继承ActionBarActivity类,并且在Manife ...

  3. 【小白的CFD之旅】06 流体力学基础

    从黄师姐那里了解到要学习CFD的话,需要先补充流体力学.数学以及计算机方面的常识,小白就一阵头大.想起当初自己已经把牛皮吹出去了,现在都不知道怎么收场,一个月入不了门多丢人.不过头大归头大,小白还是老 ...

  4. P1774 最接近神的人_NOI导刊2010[树状数组 逆序对 离散化]

    题目描述 破解了符文之语,小FF开启了通往地下的道路.当他走到最底层时,发现正前方有一扇巨石门,门上雕刻着一幅古代人进行某种活动的图案.而石门上方用古代文写着“神的殿堂”.小FF猜想里面应该就有王室的 ...

  5. cin

    cin 是预定义的标准输入流对象,cin 用来接收字符串时,遇“空格”.“TAP”.“回车”时都会结束.

  6. [备份]破解Xamarin

    [转]试用了一阵子Mono For Android,今天到期了,,囊中羞涩,只好破解. 说是要在vs2013的英文界面下运行破解包,不知道是真是假,下载并安装了一个. 然后又下载了破解包.是个名为xa ...

  7. web前端开发分享-css,js工具篇

    web前端开发乃及其它的相关开发,推荐sublime text, webstorm(jetbrains公司系列产品)这两个的原因在于,有个技术叫emmet, http://docs.emmet.io, ...

  8. Linux用户管理.md

    用户与组的概念 linux多用户,多任务的特性 Linux是一个真实的.完整的多用户多任务操作系统,多用户多任务就是可以在系统上建立多个用户,而多个用户可以在同一时间内登录同一个系统执行各自不同的任务 ...

  9. 关于refs/for/ 和refs/heads/

    1.     这个不是git的规则,而是gerrit的规则, 2.     Branches, remote-tracking branches, and tags等等都是对commite的引用(re ...

  10. 部署 OpenStack VirtualBox

    VirtualBox 中部署 OpenStack 大家新年好,CloudMan 今天给大家带来一件新年礼物. 一直以来大家都反馈 OpenStack 学习有两大障碍:1. 实验环境难搭2. 体系复杂, ...