iOS开发中GCD的一些用法

wait

wait之后的操作是在wait之前的线程任务操作完之后才进行的,哪怕wait之前的任务是异步的且放在并发队列里的(即异步执行)。
代码:

dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{
    for (NSInteger i = 0; i < 1000; i ++) {
        NSLog(@"%lu", i);
    }
    NSLog(@"waiting...");
});
dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
NSLog(@"end");

如果没有dispatch_group_wait,是如何打印的呢?end肯定是在waiting之前就打印了。
看打印:

   GCD-wait[1071:72264] 997
   GCD-wait[1071:72264] 998
   GCD-wait[1071:72264] 999
   GCD-wait[1071:72264] waitting...
   GCD-wait[1071:72065] end

打印end可以在异步执行的任务打印waiting之后进行。 使用wait会阻塞当前线程。

barrier(栅栏)

barrier的作用和wait类似,它就像一个栅栏一样拦住了同一个队列中其前面的操作,让其之后的所有操作都在其之前的所有操作完成之后才能进行。但它不会像wait一样将其他线程中的操作也拦住。
代码:

dispatch_queue_t queue = dispatch_queue_create("net.bujige.testQueue", DISPATCH_QUEUE_CONCURRENT);
// 为什么使用全局队列不行?
//    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_async(queue, ^{
    for (NSInteger i = 0; i < 1000; i ++) {
        NSLog(@"%lu", i);
    }
    NSLog(@"waiting...");
});
dispatch_barrier_async(queue, ^{
    NSLog(@"barrier");
});
dispatch_async(queue, ^{
    NSLog(@"barrier之后的任务...");
});

NSLog(@"end");

如果没有dispatch_barrier_async是如何打印的呢?“waiting”和“barrier之后的任务...”肯定是交替打印的,而且"end"会早早地就被打印了。
看打印:

 GCD-barrier[1412:136078] end
 GCD-barrier[1412:137056] 0
 GCD-barrier[1412:137056] 1
 GCD-barrier[1412:137056] 998
  
 GCD-barrier[1412:137056] 999
 GCD-barrier[1412:137056] waitting...
 GCD-barrier[1412:137056] barrier
 GCD-barrier[1412:137056] barrier之后的任务...

注意:这里的异步队列不能是全局异步队列(官方规定)

dispatch_group_notify

我们同样可以利用notify监听队列中的任务,当所有任务都执行完之后,再进行notify中的操作。

semaphore

可以将异步任务转成同步任务。
代码:

  __block NSInteger num = 0;
dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
dispatch_async(queue, ^{
    for (NSInteger i = 0; i < 1000; i ++) {
        NSLog(@"%lu", i);
        num ++;
    }
    NSLog(@"waiting...");
    dispatch_semaphore_signal(semaphore);
});
dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"num=%lu", num);
NSLog(@"end");

如果没有semaphoreend肯定会提前打印,而且打印的num会是0。
(dispatch_semaphore_signal对信号进行+1,dispatch_semaphore_wait对信号进行-1,如果信号为0,阻塞线程)
看打印:

 GCD-Semaphore[1947:219353] 997
 GCD-Semaphore[1947:219353] 998
 GCD-Semaphore[1947:219353] 999
 GCD-Semaphore[1947:219353] waiting...
 GCD-Semaphore[1947:218743] num=1000
 GCD-Semaphore[1947:218743] end

开发中如果有需要block回调之后才能进行的操作我们可以使用semaphore将代码放在block之外。

加锁
代码:

  - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
for (NSInteger i = 0; i < 100; i ++) {
    [self.mutableArr addObject:@(i)];
}
semaphoreLock = dispatch_semaphore_create(1);
dispatch_queue_t queue1 = dispatch_queue_create("queue1", DISPATCH_QUEUE_SERIAL);
dispatch_queue_t queue2 = dispatch_queue_create("queue2", DISPATCH_QUEUE_SERIAL);
dispatch_queue_t queue3 = dispatch_queue_create("queue3", DISPATCH_QUEUE_SERIAL);
dispatch_async(queue1, ^{
    [self nslog];
});
dispatch_async(queue2, ^{
    [self nslog];
});
dispatch_async(queue3, ^{
    [self nslog];
});
}
- (void)nslog{
while (1) {
    // 加锁
    dispatch_semaphore_wait(semaphoreLock, DISPATCH_TIME_FOREVER);
    if (self.mutableArr.count > 0) {
        [self.mutableArr removeObjectAtIndex:0];
        NSLog(@"%lu--%@", self.mutableArr.count, [NSThread currentThread]);
        [NSThread sleepForTimeInterval:0.2];
    }else{
        dispatch_semaphore_signal(semaphoreLock);
        break;
    }
    // 相当于解锁
    dispatch_semaphore_signal(semaphoreLock);
}
}

如果不加锁打印内容:

 GCD-线程安全[2271:260744] 3--<NSThread: 0x60000013e580>{number = 6, name = (null)}
 GCD-线程安全[2271:266057] 2--<NSThread: 0x60000013d900>{number = 4, name = (null)}
 GCD-线程安全[2271:266058] 2--<NSThread: 0x600000136740>{number = 5, name = (null)}
 GCD-线程安全[2271:260744] 0--<NSThread: 0x60000013e580>{number = 6, name = (null)}
 GCD-线程安全[2271:266057] 1--<NSThread: 0x60000013d900>{number = 4, name = (null)}

我们可以看到打印的数据混乱,如果不加锁,某一刻两个线程读到数组中的数据只剩下一个元素都进入到if判断中,其中一个线程将数组中最后一个元素移除,此时数组为空,另一个线程再对空数组做移除操作,此时就会发生崩溃,这样看来线程是不安全的。
加锁后的打印:

 GCD-线程安全[2428:278050] 4--<NSThread: 0x600003edd180>{number = 5, name = (null)}
 GCD-线程安全[2428:278048] 3--<NSThread: 0x600003e87140>{number = 3, name = (null)}
 GCD-线程安全[2428:278051] 2--<NSThread: 0x600003e85800>{number = 4, name = (null)}
 GCD-线程安全[2428:278050] 1--<NSThread: 0x600003edd180>{number = 5, name = (null)}
 GCD-线程安全[2428:278048] 0--<NSThread: 0x600003e87140>{number = 3, name = (null)}

加锁后的打印数据是正常的,并且不会由于对空数组进行操作造成程序崩溃。

本篇文章到这里就结束了,愿大家加班不多工资多,男同胞都有女朋友,女同胞都有男朋友。😊

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,193评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,306评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,130评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,110评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,118评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,085评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,007评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,844评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,283评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,508评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,395评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,985评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,630评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,797评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,653评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,553评论 2 352

推荐阅读更多精彩内容

  • 本文用来介绍 iOS 多线程中 GCD 的相关知识以及使用方法。这大概是史上最详细、清晰的关于 GCD 的详细讲...
    花花世界的孤独行者阅读 500评论 0 1
  • 本文是由于笔者在阅读有关多线程的文章的时候,看到的觉得写的很好, 就此记录下. 在开发 APP 的时候很多时候可能...
    我太难了_9527阅读 378评论 0 2
  • 1. 简介 百度百科 Grand Central Dispatch(GCD) 是 Apple 为多核的并行运算提出...
    和风细羽阅读 575评论 0 0
  • 今天来看一下iOS中的多线程开发,iOS的多线程开发也有很多种方式,不得不提的有一个就是GCD了。在介绍他之前,先...
    张囧瑞阅读 236评论 0 0
  • 原文:https://www.jianshu.com/p/2d57c72016c6[https://www.jia...
    夜雨聲煩_阅读 402评论 0 0