iOS-图文表并茂,手把手教你GCD

iOS-图文表并茂,手把手教你GCD

前言

对初学者来说,GCD似乎是一道迈不过去的坎,很多人在同步、异步、串行、并行和死锁这几个名词的漩涡中渐渐放弃治疗。本文将使用图文表并茂的方式给大家形象地解释其中的原理和规律。

线程、任务和队列的概念

异步、同步 & 并行、串行的特点

一条重要的准则

一般来说,我们使用GCD的最大目的是在新的线程中同时执行多个任务,这意味着我们需要两项条件:

  • 能开启新的线程
  • 任务可以同时执行
  • 结合以上两个条件,也就等价“开启新线程的能力 + 任务同步执行的权利”,只有在满足能力与权利这两个条件的前提下,我们才可以在同时执行多个任务。

所有组合的特点

(一)异步执行 + 并行队列

实现代码:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

//异步执行 + 并行队列

- (void)asyncConcurrent{

    //创建一个并行队列

    dispatch_queue_t queue = dispatch_queue_create("标识符", DISPATCH_QUEUE_CONCURRENT);

    NSLog(@"---start---");

    //使用异步函数封装三个任务

    dispatch_async(queue, ^{

        NSLog(@"任务1---%@", [NSThread currentThread]);

    });

    dispatch_async(queue, ^{

        NSLog(@"任务2---%@", [NSThread currentThread]);

    });

    dispatch_async(queue, ^{

        NSLog(@"任务3---%@", [NSThread currentThread]);

    });

    NSLog(@"---end---");

}

打印结果:


1

2

3

4

5

---start---

  ---end---

  任务3---{number = 5, name = (null)}

  任务2---{number = 4, name = (null)}

  任务1---{number = 3, name = (null)}

解释

    • 异步执行意味着

      • 可以开启新的线程
      • 任务可以先绕过不执行,回头再来执行
    • 并行队列意味着
      • 任务之间不需要排队,且具有同时被执行的“权利”
    • 两者组合后的结果
      • 开了三个新线程
      • 函数在执行时,先打印了start和end,再回头执行这三个任务
      • 这三个任务是同时执行的,没有先后,所以打印结果是“任务3-->任务2-->任务1”

步骤图

(二)异步执行 + 串行队列

实现代码:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

//异步执行 + 串行队列

- (void)asyncSerial{

    //创建一个串行队列

    dispatch_queue_t queue = dispatch_queue_create("标识符", DISPATCH_QUEUE_SERIAL);

    NSLog(@"---start---");

    //使用异步函数封装三个任务

    dispatch_async(queue, ^{

        NSLog(@"任务1---%@", [NSThread currentThread]);

    });

    dispatch_async(queue, ^{

        NSLog(@"任务2---%@", [NSThread currentThread]);

    });

    dispatch_async(queue, ^{

        NSLog(@"任务3---%@", [NSThread currentThread]);

    });

    NSLog(@"---end---");

}

打印结果:


1

2

3

4

5

 ---start---

 ---end---

任务1---{number = 3, name = (null)}

任务2---{number = 3, name = (null)}

任务3---{number = 3, name = (null)}

解释

  • 异步执行意味着

    • 可以开启新的线程
    • 任务可以先绕过不执行,回头再来执行
  • 串行队列意味着
    • 任务必须按添加进队列的顺序挨个执行
  • 两者组合后的结果
    • 开了一个新的子线程
    • 函数在执行时,先打印了start和end,再回头执行这三个任务
    • 这三个任务是按顺序执行的,所以打印结果是“任务1-->任务2-->任务3”

步骤图

(三)同步执行 + 并行队列

实现代码:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

//同步执行 + 并行队列

- (void)syncConcurrent{

    //创建一个并行队列

    dispatch_queue_t queue = dispatch_queue_create("标识符", DISPATCH_QUEUE_CONCURRENT);

    NSLog(@"---start---");

    //使用同步函数封装三个任务

    dispatch_sync(queue, ^{

        NSLog(@"任务1---%@", [NSThread currentThread]);

    });

    dispatch_sync(queue, ^{

        NSLog(@"任务2---%@", [NSThread currentThread]);

    });

    dispatch_sync(queue, ^{

        NSLog(@"任务3---%@", [NSThread currentThread]);

    });

    NSLog(@"---end---");

}

打印结果:


1

2

3

4

5

---start---

  任务1---{number = 1, name = main}

  任务2---{number = 1, name = main}

  任务3---{number = 1, name = main}

  ---end---

解释

  • 同步执行执行意味着

    • 不能开启新的线程
    • 任务创建后必须执行完才能往下走
  • 并行队列意味着
    • 任务必须按添加进队列的顺序挨个执行
  • 两者组合后的结果
    • 所有任务都只能在主线程中执行
    • 函数在执行时,必须按照代码的书写顺序一行一行地执行完才能继续
  • 注意事项
    • 在这里即便是并行队列,任务可以同时执行,但是由于只存在一个主线程,所以没法把任务分发到不同的线程去同步处理,其结果就是只能在主线程里按顺序挨个挨个执行了

步骤图

(四)同步执行+ 串行队列

实现代码:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

- (void)syncSerial{

    //创建一个串行队列

    dispatch_queue_t queue = dispatch_queue_create("标识符", DISPATCH_QUEUE_SERIAL);

    NSLog(@"---start---");

    //使用异步函数封装三个任务

    dispatch_sync(queue, ^{

        NSLog(@"任务1---%@", [NSThread currentThread]);

    });

    dispatch_sync(queue, ^{

        NSLog(@"任务2---%@", [NSThread currentThread]);

    });

    dispatch_sync(queue, ^{

        NSLog(@"任务3---%@", [NSThread currentThread]);

    });

    NSLog(@"---end---");

}

打印结果:


1

2

3

4

5

  ---start---

  任务1---{number = 1, name = main}

  任务2---{number = 1, name = main}

  任务3---{number = 1, name = main}

  ---end---

解释

  • 这里的执行原理和步骤图跟“同步执行+并发队列”是一样的,只要是同步执行就没法开启新的线程,所以多个任务之间也一样只能按顺序来执行,

(五)异步执行+主队列

实现代码:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

- (void)asyncMain{

    //获取主队列

    dispatch_queue_t queue = dispatch_get_main_queue();

    NSLog(@"---start---");

    //使用异步函数封装三个任务

    dispatch_async(queue, ^{

        NSLog(@"任务1---%@", [NSThread currentThread]);

    });

    dispatch_async(queue, ^{

        NSLog(@"任务2---%@", [NSThread currentThread]);

    });

    dispatch_async(queue, ^{

        NSLog(@"任务3---%@", [NSThread currentThread]);

    });

    NSLog(@"---end---");

}

打印结果:


1

2

3

4

5

  ---start---

  ---end---

  任务1---{number = 1, name = main}

  任务2---{number = 1, name = main}

  任务3---{number = 1, name = main}

解释

  • 异步执行意味着

    • 可以开启新的线程
    • 任务可以先绕过不执行,回头再来执行
  • 主队列跟串行队列的区别
    • 队列中的任务一样要按顺序执行
    • 主队列中的任务必须在主线程中执行,不允许在子线程中执行
  • 以上条件组合后得出结果:
    • 所有任务都可以先跳过,之后再来“按顺序”执行

步骤图

(六)同步执行+主队列(死锁)

实现代码:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

- (void)syncMain{

    //获取主队列

    dispatch_queue_t queue = dispatch_get_main_queue();

    NSLog(@"---start---");

    //使用同步函数封装三个任务

    dispatch_sync(queue, ^{

        NSLog(@"任务1---%@", [NSThread currentThread]);

    });

    dispatch_sync(queue, ^{

        NSLog(@"任务2---%@", [NSThread currentThread]);

    });

    dispatch_sync(queue, ^{

        NSLog(@"任务3---%@", [NSThread currentThread]);

    });

    NSLog(@"---end---");

}

打印结果:


1

  ---start---

解释

  • 主队列中的任务必须按顺序挨个执行
  • 任务1要等主线程有空的时候(即主队列中的所有任务执行完)才能执行
  • 主线程要执行完“打印end”的任务后才有空
  • “任务1”和“打印end”两个任务互相等待,造成死锁

步骤图

写在结尾的话

以上就是我对GCD的基础知识和几种组合的理解,如果觉得我的博客写得还可以,欢迎关注我的博客,本人将长期为大家推出高质量的技术博客。当然,如果觉得我哪里理解有错的,也可以留下你的评论。

时间: 2024-10-08 14:10:42

iOS-图文表并茂,手把手教你GCD的相关文章

<转>iOS-图文表并茂,手把手教你GCD

本文转自:http://www.jianshu.com/p/414b8e91e021 前言 对初学者来说,GCD似乎是一道迈不过去的坎,很多人在同步.异步.串行.并行和死锁这几个名词的漩涡中渐渐放弃治疗.本文将使用图文表并茂的方式给大家形象地解释其中的原理和规律. 线程.任务和队列的概念 异步.同步 & 并行.串行的特点 一条重要的准则 一般来说,我们使用GCD的最大目的是在新的线程中同时执行多个任务,这意味着我们需要两项条件: 能开启新的线程 任务可以同时执行 结合以上两个条件,也就等价&quo

手把手教你修改iOS版QQ的运动步数

手把手教你修改iOS版QQ的运动步数 字数669 阅读4770 评论30 喜欢33 现在很多软件都加上了运动模块,比如QQ和微信,而且还有排行榜,可以和好友比较谁的运动步数多,任何东西只要添加了比较功能,就变得不一样了.今天教大家用代码去修改QQ上的运动步数,修改完效果是这样的: 屏幕快照 2016-06-21 下午5.02.56.png 本帖子是抱着学习的心态来看待这个功能的,如果不喜欢请略过. 1准备工作 你需要一个iOS程序员所需要的设备:一个mac系统的电脑.一个Xcode.一个开发者账

【转】手把手教你把Vim改装成一个IDE编程环境(图文)

手把手教你把Vim改装成一个IDE编程环境(图文) By: 吴垠 Date: 2007-09-07 Version: 0.5 Email: lazy.fox.wu#gmail.com Homepage: http://blog.csdn.net/wooin Copyright: 该文章版权由吴垠和他可爱的老婆小包子所有.可在非商业目的下任意传播和复制.对于商业目的下对本文的任何行为需经作者同意.联系方式:lazy.fox.wu#gmail.com 1 写在前面   Linux下编程一直被诟病的一

只需 4 步,手把手教你如何实现滤镜功能

源地址:http://zihua.li/2014/06/implement-instagram-like-filters/ 滤镜对于照片而言,起到的是雪中送炭和锦上添花的作用.优秀的滤镜,能让随手之作显得别有风味,又能为已经绝色的照片画龙点睛.现在几乎凡是和照片相关的应用程序都有滤镜功能,而相较而言介绍滤镜原理和实现的文章又少之又少,为此我专门写了这篇文章来系统地讲解滤镜是如何实现的. 什么是滤镜 滤镜最初是指安装在相机镜头前过滤自然光的附加镜头,用来实现调色和添加效果.一些数字图像处理软件(最

深入理解iPhone数据持久化(手把手教你iphone开发 – 基础篇)

在所有的移动开发平台数据持久化都是很重要的部分:在j2me中是rms或保存在应用程序的目录中,在symbian中可以保存在相应的磁盘目录中和数据库中.symbian中因为权限认证的原因,在3rd上大多数只能访问应用程序的private目录或其它系统共享目录.在iphone中,apple博采众长,提供了多种数据持久化的方法,下面笔者会逐个进行详细的讲解. iphone提供的数据持久化的方法,从数据保存的方式上讲可以分为三大部分:属性列表.对象归档.嵌入式数据库(SQLite3).其他方法. 一.属

手把手教你写专利申请书/怎样申请专利

手把手教你写专利申请书·怎样申请专利 摘要小前言(一)申请前的准备工作    1.申请前查询    2.其它方面的考虑    3.申请文件准备(二)填写专利申请系列文档    1.实际操作步骤    2.详细操作    3.经验分享.注意事项(三)关于费用(四)其它的话參考资源提示常见问题的问与答 摘要: 怎样写好专利申请?由于非常多专利申请人都是第一次申请,因此,可能有一种神奇和些许恐惧.本文写的是怎样写专利申请书,手把手教你写专利申请并提供申请专利时的注意事项,专利申请费用及费用减缓等相关參

手把手教你ranorex_android自动化测试第一个示例

要说android的自动化,那真是折腾死我了,从早期的monkeyrunner,到后来的robotium,再到最新的uiautomator,各有各的问题,总之性价比都不够高,不太适合我的使用场景.于是不愿意将研究结果投入实践,一直等待.今年的适配问题越发突出,再不自动化,回归就该搞死人了. 于是最后研究比对instrumetation和uiautomator,终于确定了uiautomator方案,其实也是勉为其难,不过总算也是可以实现的.先准备好脚本,批量修改了所有布局文件,增加了content

CSS3实战开发:手把手教大家搜索表单发光特效实战开发

各位网友,大家好,我是陌上花会开,今天这个案例,我将带领大家,手把手教大家开发一个会发光的表单组,还是先秀运行效果图吧: 从这个效果图,大家可以发现,当光标在某个组件上时,它就会发光,今天我将带领大家实战开发这个特效. PS: 大家可以直接将博客中的代码复制到本地便可得到同样的运行效果. 首先,我们先准备好html代码: <!doctype html> <html> <head> <meta charset="utf-8"><met

手把手教你安装Hbase,一次成功!

安装环境: OS: Centos 6.5 JDK: jdk1.6.0_18 Hadoop: hadoop-0.20.2 Hbase: hbase-0.90.5 安装准备: 1.       Jdk环境已安装:版本为1.6以上 2.       hadoop环境已安装:完全分布模式安装如下 http://blog.csdn.net/lichangzai/article/details/8206834 3.       hbase版本选择 Hbase 版本必需与 Hadoop版本匹配,否则会安装失败