iOS 实现所有网络请求都返回再处理回调

下面实现3种功能:

  • 同时发起请求,乱序返回,全部请求都返回后再统一回调。
  • 同时发起请求,有序返回,全部请求都返回后再统一回调。
  • 逐一发起请求,等待请求返回后再发起下一个请求,全部请求都返回再统一回调。

要自己去理解这些概念

  • 串行队列:DISPATCH_QUEUE_SERIAL
  • 并行队列:DISPATCH_QUEUE_CONCURRENT
  • 同步:dispatch_sync
  • 异步:dispatch_async

主要代码一:

1、dispatch_group_t downloadGroup = dispatch_group_create();
2、dispatch_group_enter(downloadGroup);
3、dispatch_group_leave(downloadGroup);
4、dispatch_group_notify(downloadGroup, dispatch_get_main_queue(), ^{
});

主要代码二:

1、dispatch_queue_t queue = dispatch_queue_create("CONCURRENT_QUEUE", DISPATCH_QUEUE_CONCURRENT);
2、dispatch_async(queue, ^{
});

主要代码三:

初始化一个NSConditionLock对象,锁条件为0
1、NSConditionLock *lock = [[NSConditionLock alloc] initWithCondition:0];
当i满足锁条件时加锁
2、[lock lockWhenCondition:i];
解锁,并重置锁的条件为i+1
3、[lock unlockWithCondition:i+1];

测试数据

- (NSArray *)dataArray {
    return @[
             @"http://ww3.sinaimg.cn/mw690/8245bf01jw1f2jhh2ohanj20jg0yk418.jpg",
             @"http://www.pianshen.com/images/485/c0f2488fb65f3fdc25e073cd1fcee8b5.JPEG",//大图,加载慢
             @"http://ww1.sinaimg.cn/mw690/8245bf01jw1f2jhh3grfwj20jg0pxn13.jpg",
             @"http://www.pianshen.com/images/786/0d4cbd92ce5209cfd0aadd6c6cc5bdca.JPEG",//大图,加载慢
             @"http://www.pianshen.com/images/742/00717c2ca89c5b1e6ab4ff6c568ed93e.JPEG",//大图,加载慢
             @"http://ww2.sinaimg.cn/mw690/8245bf01jw1f2jhh4mutgj20jg0ly0xt.jpg",
             @"http://ww3.sinaimg.cn/mw690/8245bf01jw1f2jhh4vc7pj20jg0px41m.jpg",
             @"http://ww4.sinaimg.cn/mw690/8245bf01jw1f2jhh2mgkgj20jg0pxn2z.jpg"
             ];
}

1、并行请求,乱序处理返回,全部返回完成后再回调

测试代码

- (void)one {
    /// 1、组合队列
    dispatch_group_t downloadGroup = dispatch_group_create();
    NSArray *array = [self dataArray];
    for (int i = 0; i < array.count; i++) {
        NSLog(@"请求--- 前 %d",i);
        /// 2、Enter
        dispatch_group_enter(downloadGroup);
        NSURL *URL = [NSURL URLWithString:array[i]];
        NSURLRequest *request = [NSURLRequest requestWithURL:URL];
        NSURLSession *session = [NSURLSession sharedSession];
        NSURLSessionDataTask *task = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
            NSLog(@"请求--- 后 %d",i);
            /// 3、Leave
            dispatch_group_leave(downloadGroup);
        }];
        [task resume];
    }
    /// 结束统一处理
    dispatch_group_notify(downloadGroup, dispatch_get_main_queue(), ^{
        NSLog(@"请求--- 结束统一处理");
    });
}

测试结果

[1687:259031] 请求--- 前 0
[1687:259031] 请求--- 前 1
[1687:259031] 请求--- 前 2
[1687:259031] 请求--- 前 3
[1687:259031] 请求--- 前 4
[1687:259031] 请求--- 前 5
[1687:259031] 请求--- 前 6
[1687:259031] 请求--- 前 7

[1687:259076] 请求--- 后 0
[1687:259076] 请求--- 后 2
[1687:259076] 请求--- 后 3
[1687:259076] 请求--- 后 5
[1687:259076] 请求--- 后 6
[1687:259076] 请求--- 后 7
[1687:259077] 请求--- 后 1
[1687:259077] 请求--- 后 4
[1687:259031] 请求--- 结束统一处理

2、并行请求,有序处理返回,全部返回完成后再回调

注意:有序代码要放在两者之间:
[lock lockWhenCondition:i];
// 这里写有序代码
[lock unlockWithCondition:i+1];

测试代码

- (void)two {
    /// 1、组合队列 + 加条件锁
    dispatch_group_t downloadGroup = dispatch_group_create();
    dispatch_queue_t queue = dispatch_queue_create("CONCURRENT_QUEUE", DISPATCH_QUEUE_CONCURRENT);
    NSConditionLock *lock = [[NSConditionLock alloc] initWithCondition:0];
    
    NSArray *array = [self dataArray];
    for (int i = 0; i < array.count; i++) {
        /// 2、Enter
        dispatch_group_enter(downloadGroup);
        NSLog(@"请求--- 前 %d",i);
        
        NSURL *URL = [NSURL URLWithString:array[i]];
        NSURLRequest *request = [NSURLRequest requestWithURL:URL];
        NSURLSession *session = [NSURLSession sharedSession];
        NSURLSessionDataTask *task = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
            dispatch_async(queue, ^{
                [lock lockWhenCondition:i];
                // 有序代码要放在这里,记得是这里
                NSLog(@"请求--- 后 %d",i);
                /// 3、Leave
                dispatch_group_leave(downloadGroup);
                [lock unlockWithCondition:i+1];
            });
        }];
        [task resume];
    }
    /// 结束统一处理
    dispatch_group_notify(downloadGroup, dispatch_get_main_queue(), ^{
        NSLog(@"请求--- 结束统一处理");
    });
}

或者

- (void)two1 {
    /// 1、组合队列 + 信号通知
    dispatch_group_t group = dispatch_group_create();
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
    dispatch_queue_t queue = dispatch_queue_create(NULL, DISPATCH_QUEUE_SERIAL);
    
    NSArray *array = [self testArray];
    for (int i = 0; i < array.count; i++) {
        NSLog(@"请求--- 前 %d",i);
        dispatch_group_async(group, queue, ^{
            NSURL *URL = [NSURL URLWithString:array[i]];
            NSURLRequest *request = [NSURLRequest requestWithURL:URL];
            NSURLSession *session = [NSURLSession sharedSession];
            NSURLSessionDataTask *task = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
                NSLog(@"请求--- 后 %d",i);
                dispatch_semaphore_signal(semaphore);
            }];
            [task resume];
            dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
        });
    }
    /// 结束统一处理
    dispatch_group_notify(group, queue, ^{
        NSLog(@"请求--- 结束统一处理");
    });
}

测试结果

[1745:263395] 请求--- 前 0
[1745:263395] 请求--- 前 1
[1745:263395] 请求--- 前 2
[1745:263395] 请求--- 前 3
[1745:263395] 请求--- 前 4
[1745:263395] 请求--- 前 5
[1745:263395] 请求--- 前 6
[1745:263395] 请求--- 前 7

[1745:263422] 请求--- 后 0
[1745:263447] 请求--- 后 1
[1745:263422] 请求--- 后 2
[1745:263432] 请求--- 后 3
[1745:263447] 请求--- 后 4
[1745:263437] 请求--- 后 5
[1745:263436] 请求--- 后 6
[1745:263446] 请求--- 后 7
[1745:263395] 请求--- 结束统一处理

3、串行请求,等着一个请求有返回之后才请求下一个,有序处理返回,全部返回完成后再回调

测试代码

- (void)three {
    /// 1、组合队列 + 加条件锁
    dispatch_group_t downloadGroup = dispatch_group_create();
    dispatch_queue_t queue = dispatch_queue_create("CONCURRENT_QUEUE", DISPATCH_QUEUE_CONCURRENT);
    NSConditionLock *lock = [[NSConditionLock alloc] initWithCondition:0];
    
    NSArray *array = [self dataArray];
    for (int i = 0; i < array.count; i++) {
        /// 2、Enter
        dispatch_group_enter(downloadGroup);
        dispatch_async(queue, ^{
            [lock lockWhenCondition:i];
            NSLog(@"请求--- 前 %d",i);
            
            NSURL *URL = [NSURL URLWithString:array[i]];
            NSURLRequest *request = [NSURLRequest requestWithURL:URL];
            NSURLSession *session = [NSURLSession sharedSession];
            NSURLSessionDataTask *task = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
                // 有序代码要放在这里,记得是这里
                NSLog(@"请求--- 后- %d",i);
                /// 3、Leave
                dispatch_group_leave(downloadGroup);
                [lock unlockWithCondition:i+1];
            }];
            [task resume];
        });
    }
    /// 结束统一处理
    dispatch_group_notify(downloadGroup, dispatch_get_main_queue(), ^{
        NSLog(@"请求--- 结束统一处理");
    });
}

测试结果

2019-05-17 13:11:22.968090+0800 [1933:279351] 请求--- 前 0
2019-05-17 13:11:22.976308+0800 [1933:279369] 请求--- 后- 0
2019-05-17 13:11:22.976597+0800 [1933:279345] 请求--- 前 1
2019-05-17 13:11:23.004162+0800 [1933:279372] 请求--- 后- 1
2019-05-17 13:11:23.004598+0800 [1933:279356] 请求--- 前 2
2019-05-17 13:11:23.009802+0800 [1933:279356] 请求--- 后- 2
2019-05-17 13:11:23.010347+0800 [1933:279342] 请求--- 前 3
2019-05-17 13:11:23.020183+0800 [1933:279372] 请求--- 后- 3
2019-05-17 13:11:23.020621+0800 [1933:279355] 请求--- 前 4
2019-05-17 13:11:23.027247+0800 [1933:279355] 请求--- 后- 4
2019-05-17 13:11:23.027687+0800 [1933:279368] 请求--- 前 5
2019-05-17 13:11:23.032935+0800 [1933:279371] 请求--- 后- 5
2019-05-17 13:11:23.033352+0800 [1933:279351] 请求--- 前 6
2019-05-17 13:11:23.038668+0800 [1933:279371] 请求--- 后- 6
2019-05-17 13:11:23.038819+0800 [1933:279370] 请求--- 前 7
2019-05-17 13:11:23.053150+0800 [1933:279373] 请求--- 后- 7
2019-05-17 13:11:23.122767+0800 [1933:279309] 请求--- 结束统一处理
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容