iOS:GCD【dispatch_semaphore】使用之所思

最近在使用GCD的时候,发现自己对dispatch_semaphore理解不是那么深刻,所以自己在网上找资料学习dispatch_semaphore,并对dispatch_semaphore有了一些自己的看法和理解。

1.异步任务使用semaphore

    NSLog(@"----------------------------开始----------------------------  线程:%@",[NSThread currentThread]);
    dispatch_semaphore_t sema =  dispatch_semaphore_create(0);
    
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"---------------------------- 任务一开始 ---------------------------- 线程:%@",[NSThread currentThread]);
        [NSThread sleepForTimeInterval:2.0];
        NSLog(@"---------------------------- 任务一结束 ---------------------------- 线程:%@",[NSThread currentThread]);
        dispatch_semaphore_signal(sema);
    });
    dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
    
    
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"---------------------------- 任务二开始 ---------------------------- 线程:%@",[NSThread currentThread]);
        [NSThread sleepForTimeInterval:2.0];
        NSLog(@"---------------------------- 任务二结束 ---------------------------- 线程:%@",[NSThread currentThread]);
        dispatch_semaphore_signal(sema);
    });
    dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
    
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"---------------------------- 任务三开始 ---------------------------- 线程:%@",[NSThread currentThread]);
        [NSThread sleepForTimeInterval:2.0];
        NSLog(@"---------------------------- 任务三结束 ---------------------------- 线程:%@",[NSThread currentThread]);
        dispatch_semaphore_signal(sema);
    });
    dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
    
    NSLog(@"----------------------------结束---------------------------- 线程:%@",[NSThread currentThread]);

打印结果如下:

2018-11-20 10:05:48.165624+0800 Demo[2920:724653] ----------------------------开始----------------------------  线程:<NSThread: 0x282c6b180>{number = 1, name = main}
2018-11-20 10:05:48.166227+0800 Demo[2920:724705] ---------------------------- 任务一开始 ---------------------------- 线程:<NSThread: 0x282cebc40>{number = 3, name = (null)}
2018-11-20 10:05:50.171470+0800 Demo[2920:724705] ---------------------------- 任务一结束 ---------------------------- 线程:<NSThread: 0x282cebc40>{number = 3, name = (null)}
2018-11-20 10:05:50.171953+0800 Demo[2920:724705] ---------------------------- 任务二开始 ---------------------------- 线程:<NSThread: 0x282cebc40>{number = 3, name = (null)}
2018-11-20 10:05:52.177195+0800 Demo[2920:724705] ---------------------------- 任务二结束 ---------------------------- 线程:<NSThread: 0x282cebc40>{number = 3, name = (null)}
2018-11-20 10:05:52.177657+0800 Demo[2920:724710] ---------------------------- 任务三开始 ---------------------------- 线程:<NSThread: 0x282ce9380>{number = 4, name = (null)}
2018-11-20 10:05:54.182973+0800 Demo[2920:724710] ---------------------------- 任务三结束 ---------------------------- 线程:<NSThread: 0x282ce9380>{number = 4, name = (null)}
2018-11-20 10:05:54.183224+0800 Demo[2920:724653] ----------------------------结束---------------------------- 线程:<NSThread: 0x282c6b180>{number = 1, name = main}

这里我们可以分析到,semaphore确实起到了阻塞等待的作用。dispatch_semaphore_t sema = dispatch_semaphore_create(0)首先创建了一个信号量为0的semaphore,接着开启了一个异步任务,在任务中dispatch_semaphore_signal(sema)会使信号量+1,然后在主线程中dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER)使信号量-1,此时信号量为-1,那么会一直等待。所以主线程会一直等待,直到异步任务执行完毕dispatch_semaphore_signal(sema)这里,对信号量+1,然后才会接着往下走。后面的任务以此类推。

可是在我们平常的使用中,更多是配合网络请求。例如请求B的数据依赖于请求A,只有先拿到了请求A的数据,才能请求B,当然这个需求解决办法有很多,但这里我们就只谈谈怎么用semaphore来解决这个需求。接着我们利用相同思路,结合AFN使用semaphore:

NSLog(@"----------------- 测试开始 -----------------");
    NSDictionary *dict = @{ @"page": @1,
                            @"pageSize": @20};
    
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
    //对AFN的简单封装 请求一个post任务
    [[CYW_NetworkingManager shareManager] cyw_networkType:NetWorkTypePOST
                                            withURLString:@"https://api.it120.cc/tz/shop/goods/list" withParameters:dict
                                         withSuccessBlock:^(id result) {
                                             NSLog(@"----------------- 任务一完成 -----------------%@ -----------------",[NSThread currentThread]);
                                             dispatch_semaphore_signal(semaphore);
                                             
                                         } withFailBlock:^(NSError *error) {
                                             dispatch_semaphore_signal(semaphore);
                                         }];
    dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    //请求一个get任务
    [[CYW_NetworkingManager shareManager] cyw_networkType:NetWorkTypeGET withURLString:@"https://api.it120.cc/tz/shop/goods/category/all" withParameters:nil withSuccessBlock:^(id result) {
        
        NSLog(@"----------------- 任务二完成 ----------------- %@ -----------------",[NSThread currentThread]);
        dispatch_semaphore_signal(semaphore);

    } withFailBlock:^(NSError *error) {
        dispatch_semaphore_signal(semaphore);
    }];
    dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
 
    NSLog(@"----------------- 测试结束 -----------------");

打印结果如下:

2018-11-20 10:05:01.794640+0800 Demo[2918:724320] ----------------- 测试开始 -----------------

这里我们发现,仅仅只打印了测试开始,后面的任务都没有执行。那么究竟是什么原因,导致了主线程的阻塞呢?原来在AFN中,AFN已经已经将回调回到了主线程中:

  [[CYW_NetworkingManager shareManager] cyw_networkType:NetWorkTypeGET withURLString:@"https://api.it120.cc/tz/shop/goods/category/all" withParameters:nil withSuccessBlock:^(id result) {
        NSLog(@"----------------- 当前线程:%@ -----------------",[NSThread currentThread]);

    } withFailBlock:^(NSError *error) {
        
    }];

打印结果如下:
2018-11-20 10:11:32.721403+0800 Demo[2924:725812] ----------------- 当前线程:<NSThread: 0x283e4fa00>{number = 1, name = main} -----------------

而在执行到第一个dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER)时候,信号量就为0了,将主线程阻塞了,就没有办法继续往下执行dispatch_semaphore_signal(semaphore)。那为什么上面第一种方法可以执行呢?是因为上面的任务都是异步的,所以就算主线程阻塞了,在异步任务中还是可以执行dispatch_semaphore_signal(semaphore),也就不存在阻塞问题了。
那么我们怎么去解决这样的问题呢?这里我们就需要用到GCD的另外一种使用方式了:dispatch_group。对dispatch_group不是很了解的小伙伴可以参考这里,里面对GCD的各种使用方法都有很详细的讲解。
好了,废话不多说,接下来我们看看怎么用dispatch_group完成dispatch_semaphore和AFN的结合使用:

2.结合AFN使用dispatch_semaphore

NSLog(@"----------------- 测试开始 -----------------");
    dispatch_group_t gruop = dispatch_group_create();
    dispatch_group_async(gruop, dispatch_get_global_queue(0, 0), ^{
        dispatch_semaphore_t sema = dispatch_semaphore_create(0);
        NSLog(@"----------------- gruop:%@ -----------------",[NSThread currentThread]);
        
        [[CYW_NetworkingManager shareManager] cyw_networkType:NetWorkTypeGET withURLString:@"https://api.it120.cc/tz/shop/goods/category/all" withParameters:nil withSuccessBlock:^(id result) {
            
            NSLog(@"----------------- 任务一完成 ----------------- %@ -----------------",[NSThread currentThread]);
            dispatch_semaphore_signal(sema);
            
        } withFailBlock:^(NSError *error) {
            dispatch_semaphore_signal(sema);
        }];
        dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
        
        
        [[CYW_NetworkingManager shareManager] cyw_networkType:NetWorkTypeGET withURLString:@"https://api.it120.cc/tz/shop/goods/category/all" withParameters:nil withSuccessBlock:^(id result) {
            
            NSLog(@"----------------- 任务二完成 ----------------- %@ -----------------",[NSThread currentThread]);
            dispatch_semaphore_signal(sema);
            
        } withFailBlock:^(NSError *error) {
            dispatch_semaphore_signal(sema);
        }];
        dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
        
        
        [[CYW_NetworkingManager shareManager] cyw_networkType:NetWorkTypeGET withURLString:@"https://api.it120.cc/tz/shop/goods/category/all" withParameters:nil withSuccessBlock:^(id result) {
            
            NSLog(@"----------------- 任务三完成 ----------------- %@ -----------------",[NSThread currentThread]);
            dispatch_semaphore_signal(sema);
            
        } withFailBlock:^(NSError *error) {
            dispatch_semaphore_signal(sema);
        }];
        dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
        
    });
    
    dispatch_group_notify(gruop, dispatch_get_main_queue(), ^{
       NSLog(@"----------------- 测试结束 -----------------");
    });

打印结果如下:

2018-11-20 10:31:07.634295+0800 Demo[2942:730510] ----------------- 测试开始 -----------------
2018-11-20 10:31:07.636867+0800 Demo[2942:730582] ----------------- gruop:<NSThread: 0x283c8ebc0>{number = 3, name = (null)} -----------------
2018-11-20 10:31:08.057232+0800 Demo[2942:730510] ----------------- 任务一完成 ----------------- <NSThread: 0x283c01b40>{number = 1, name = main} -----------------
2018-11-20 10:31:08.130937+0800 Demo[2942:730510] ----------------- 任务二完成 ----------------- <NSThread: 0x283c01b40>{number = 1, name = main} -----------------
2018-11-20 10:31:08.229891+0800 Demo[2942:730510] ----------------- 任务三完成 ----------------- <NSThread: 0x283c01b40>{number = 1, name = main} -----------------
2018-11-20 10:31:08.230509+0800 Demo[2942:730510] ----------------- 测试结束 -----------------

根据打印的结果,这里完美解决了A任务、B任务之间的依赖关系的需求。也再次证明AFN的回调确实在主线程中(AFN的源码中,也可以找到,有兴趣的小伙伴可以研究下源码)。
那么,为什么用到dispatch_group可以解决问题呢?这里dispatch_group_async(gruop, dispatch_get_global_queue(0, 0)开辟一个新的子线程。而在子线程中,dispatch_semaphore_t sema = dispatch_semaphore_create(0)创建了一个信号量为0的semaphore,紧接着在当前子线程中执行dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER),那么会阻塞当前线程,一直等待,直到等待AFN的回调中dispatch_semaphore_signal(sema)将信号量+1。

如果对以上理解了的话,同样开启一个异步任务,也可以将任务一、二、三按顺序完成:

    NSLog(@"----------------- 测试开始 -----------------");
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        
        dispatch_semaphore_t sema = dispatch_semaphore_create(0);
        
        [[CYW_NetworkingManager shareManager] cyw_networkType:NetWorkTypeGET withURLString:@"https://api.it120.cc/tz/shop/goods/category/all" withParameters:nil withSuccessBlock:^(id result) {
            
            NSLog(@"----------------- 任务一完成 ----------------- %@ -----------------",[NSThread currentThread]);
            dispatch_semaphore_signal(sema);
            
        } withFailBlock:^(NSError *error) {
            dispatch_semaphore_signal(sema);
        }];
        dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
        
        
        [[CYW_NetworkingManager shareManager] cyw_networkType:NetWorkTypeGET withURLString:@"https://api.it120.cc/tz/shop/goods/category/all" withParameters:nil withSuccessBlock:^(id result) {
            
            NSLog(@"----------------- 任务二完成 ----------------- %@ -----------------",[NSThread currentThread]);
            dispatch_semaphore_signal(sema);
            
        } withFailBlock:^(NSError *error) {
            dispatch_semaphore_signal(sema);
        }];
        dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
        
        
        [[CYW_NetworkingManager shareManager] cyw_networkType:NetWorkTypeGET withURLString:@"https://api.it120.cc/tz/shop/goods/category/all" withParameters:nil withSuccessBlock:^(id result) {
            
            NSLog(@"----------------- 任务三完成 ----------------- %@ -----------------",[NSThread currentThread]);
            dispatch_semaphore_signal(sema);
            
        } withFailBlock:^(NSError *error) {
            dispatch_semaphore_signal(sema);
        }];
        dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
        
    });
    NSLog(@"----------------- 测试结束 -----------------");


打印结果如下:

2018-11-20 10:38:48.473069+0800 Demo[2950:732087] ----------------- 测试开始 -----------------
2018-11-20 10:38:48.473251+0800 Demo[2950:732087] ----------------- 测试结束 -----------------
2018-11-20 10:38:48.917159+0800 Demo[2950:732087] ----------------- 任务一完成 ----------------- <NSThread: 0x282351b40>{number = 1, name = main} -----------------
2018-11-20 10:38:49.005818+0800 Demo[2950:732087] ----------------- 任务二完成 ----------------- <NSThread: 0x282351b40>{number = 1, name = main} -----------------
2018-11-20 10:38:49.092094+0800 Demo[2950:732087] ----------------- 任务三完成 ----------------- <NSThread: 0x282351b40>{number = 1, name = main} -----------------

这里可以看到,虽然并不像使用dispatch_gruop那样,将调度组中的任务,执行完毕后再去调用
dispatch_group_notify里面的任务。但任务一、二、三却也是按顺序执行的。

以上是我个人对dispatch_semaphore的一点点理解,有错误的地方希望大家能指出来。

最后,个人还有点小疑问。既然AFN的回调已经在主线程中了,那么为什么我们还要AFN的回调中用dispatch_get_main_queue()回到主线程更新UI呢?

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

推荐阅读更多精彩内容