多个网络请求的接口设计

最近做到一个页面里有四个接口,每一个接口返回都有对应的UI改变,所以要四个接口全部请求返回成功后,再去做UI刷新。然而由于网络请求用的是AFN等第三方库,本身就是异步的,所以GCD的栅栏函数dispatch_barrier_async是没有用的。

那么就让他一个一个的执行,等到所有都执行完成后再去刷新UI,于是乎呢,可以在第一个网络请求了的返回block里去做第二个网络请求,以此类推就可以完成最后的UI刷新。这个不失为一种比较有效的方法,但是这样写的代码真的挺丑的。。。。

如果单纯的想让这几个网络请求按顺序执行,先进先出,那就是队列啊,iOS里提供的队列管理就是GCD这个强大的c语言实现的库。GCD里面有一个信号量--dispatch_semaphore_t,

dispatch_semaphore_t

dispatch_semaphore_create(long value);创建一个信号量,如果value小于0的话,这个信号量就是nil。
dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatch_time_t timeout);可以让信号量减1,如果信号量是0,那么他会等待信号量为非零才会执行下一步。
dispatch_semaphore_signal(dispatch_semaphore_t dsema);可以让信号量增加1.

通过这上面的方法,我们可以一开始设计信号量是0,每一个网络请求之前用dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatch_time_t timeout);,请求结束回调里用dispatch_semaphore_signal(dispatch_semaphore_t dsema);这样可以让请求按照顺序执行下来。Talk is cheep,show your the code:

- (void)semaphoretest {
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);

    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        sleep(2);
        NSLog(@"任务1完成---- %@", [NSThread currentThread]);
        dispatch_semaphore_signal(semaphore);
    });

    dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        sleep(2);
        NSLog(@"任务2完成---- %@", [NSThread currentThread]);
        dispatch_semaphore_signal(semaphore);
    });

    dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        sleep(2);
        NSLog(@"任务3完成---- %@", [NSThread currentThread]);
        dispatch_semaphore_signal(semaphore);
    });
}

打印结果如下:

信号量结果.png

很清楚的看到了他们是按照顺序执行的。

dispatch_group

最好的设计当然是他们几个并发执行,到时候全部请求结束后再去做UI的刷新。于是乎想到了CGD的group啦!GCD就是辣么强大有木有!设计如下:

- (void)groupTest {
    dispatch_queue_t quete = dispatch_queue_create("XIAXIAQUEUE", DISPATCH_QUEUE_CONCURRENT);
    dispatch_group_t group = dispatch_group_create();

    dispatch_group_enter(group);
    dispatch_group_async(group, quete, ^{
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
            sleep(5);
        
            NSLog(@"网络任务1完----%@", [NSThread currentThread]);
            dispatch_group_leave(group);
        });
        sleep(2);
        NSLog(@"任务1完成---%@", [NSThread currentThread]);
    });

//    dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
    dispatch_group_enter(group);
    dispatch_group_async(group, quete, ^{
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
            sleep(5);
        
            NSLog(@"网络任务2完成----%@", [NSThread currentThread]);
            dispatch_group_leave(group);
        });
        sleep(2);
        NSLog(@"任务2完成---%@", [NSThread currentThread]);
    
    });

//    dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
    dispatch_group_enter(group);
    dispatch_group_async(group, quete, ^{
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
            sleep(5);
        
            NSLog(@"网络任务3完成 --- %@", [NSThread currentThread]);
            dispatch_group_leave(group);
        });
        sleep(2);
        NSLog(@"任务3完成---- %@", [NSThread currentThread]);
    });



    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
        NSLog(@"最后执行 ---- %@", [NSThread currentThread]);
    });
}

打印的结果如下:

group结果.png

首先可以发现其实这几个任务都是在不同的线程的,但是最后执行的肯定是最后一步!

结束语

作为一个程序猿,思考是最重要的,虽然现在网上很多代码可以用拿来主义,但是如果只会拿来,那只能作为一个初级的程序猿了。所以有时候多思考,多想想。

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

推荐阅读更多精彩内容

  • 谈到iOS多线程,一般都会谈到四种方式:pthread、NSThread、GCD和NSOperation。其中,苹...
    攻城狮GG阅读 2,299评论 0 3
  • 我们在开发中可能会遇到多个网络请求的需求,比如一个界面有两个请求,tableView的数据源及表头,需要等两个请求...
    CholMay阅读 15,221评论 20 78
  • 上两篇文章我们针对椎间盘突出症做了详细探讨,不断有同学咨询关于营养学上椎间盘突出症的辅助食补疗法,今天我们就来深入...
    依依营养学院阅读 4,037评论 0 2
  • 我自信命运掌握在自己手中 原来上天早有安排 既然结局已定 那又何须执着 人生本来就是一个圈,不管走多远,走多快,最...
    龙之君临天下阅读 3,904评论 0 0
  • 朋友之前参加了一个健康讲座,一位专家提到了一点:一天抽5根烟对身体基本无害。这位好友便听了进去,保持着每天5根烟的...
    大东教练阅读 9,348评论 0 0