GCD深入理解学习笔记

1.在重载UIViewControll的viewDidLoad时不能加入太多的工作,这样会引起视图控制器出现前等待时间过长。尽量把一些工作放到后台,如果它们不是绝对必须要运行在加载时间里。

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{
   dispatch_async(dispatch_get_main_queue(), ^{
   });
 });

2.单例的线程安全

  • 这种方法线程不安全

竞态条件:当线程A进入到if语句块并在sharedPhotoManager被分配内存前发生一个上下文切换(比如[NSThread sleepForTimeInterval:2])。然后另一个线程B可能进入if,分配单例实例的内存,然后退出。当系统上下文切换回线程A,会分配另外一个单例实例的内存,然后退出。这时,就有了两个单例的实例!这时显然就不是单例了!

+ (instancetype)sharedManager 
{ static PhotoManager *sharedPhotoManager = nil; 
 if (!sharedPhotoManager)
 {  
  sharedPhotoManager = [[self alloc] init]; 
 }
 return sharedPhotoManager;
}

dispatch_once()以线程安全的方式只执行其代码块一次,试图访问临界区(即传递给dispatch_once的代码)的不同线程会在临界区已有一个线程的情况下被阻塞,直到临界区完成为止。
*Critical Section 临界区:两个线程不能同时执行这段代码,一段代码不能被并发执行 *

+ (instancetype)sharedManager
{ 
 static PhotoManager *sharedPhotoManager = nil;
 static dispatch_once_t onceToken; dispatch_once(&onceToken,^{
 sharedPhotoManager = [[self alloc] init]; 
 }); 
return sharedPhotoManager;
}

3.读写线程安全
众多的Foundation 容器类不是线程安全的。当一个线程正在读取时,另一个线程修改该容器类此时不是线程安全的。
Dispatch barriers 是一组函数,在并发队列上扮演一个串行的瓶颈。使用GCD的障碍API确保提交的Block在那个特定时间上是指定队列上唯一被执行的条目。这就意味着所有先于调度障碍提交到队列的条目必能在这个Block执行前完成。

barriers.png

注意到正常部分的操作就如同一个正常的并发队列。但当障碍执行时,它本质上就如同一个串行队列。也就是,障碍是唯一在执行的事物。在障碍完成后,队列回到一个正常并发队列的样子。
只在自定义并发队列中使用
在执行写操作时

写操作.png
读操作.png
实例化.png

本文内容来自

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • GCD 深入理解:第一部分 什么是 GCD GCD 是 libdispatch 的市场名称,而 libdispat...
    willphonez阅读 659评论 0 2
  • 虽然 GCD 已经出现过一段时间了,但不是每个人都明了其主要内容。这是可以理解的;并发一直很棘手,而 GCD 是基...
    随风飘荡的小逗逼阅读 1,397评论 0 2
  • 本文翻译自 http://www.raywenderlich.com/60749/grand-central-di...
    skogt阅读 1,260评论 2 10
  • 烤花生准备做牛轧糖。原来牛轧糖是nougat的音译,和牛没半毛钱关系,而包糖的糯米纸,既不是纸,也不是糯米做的。中...
    momo2008阅读 252评论 0 0
  • 把我想说的话混在风中,掺在雨里,让春风带给你,讲给爱淋雨的你听。 这种只有我们能听懂的语言,不怕别人偷走。 风说喜...
    齐天章阅读 258评论 1 1