信号量dispatch_semaphore在iOS APP编程里的“应用”

来源:互联网 发布:手机五线谱制作软件 编辑:程序博客网 时间:2024/05/21 03:29

信号量概述(参考百度百科):

  以一个停车场的运作为例。简单起见,假设停车场只有三个车位,一开始三个车位都是空的。这时如果同时来了五辆车,看 门人允许其中三辆直接进入,然后放下车拦,剩下的车则必须在入口等待,此后来的车也都不得不在入口处等待。这时,有一辆车离开停车场,看门人得知后,打开车拦,放入外面的一辆进去,如果又离开两辆,则又可以放入两辆,如此往复。在这个停车场系统中,车位是公共资源,每辆车好比一个线程,看门人起的就是信号量的作用。关于iOS多线程,需要深入了解的可以,查看王坤教学视频

  抽象的来讲,信号量的特性如下:信号量是一个非负整数(车位数),所有通过它的线程/进程(车辆)都会将该整数减一(通过它当然是为了使用资源),当该整数值为零时,所有试图通过它的线程都将处于等待状态。在信号量上我们定义两种操作: Wait(等待) 和 Release(释放)。当一个线程调用Wait操作时,它要么得到资源然后将信号量减一,要么一直等下去(指放入阻塞队列),直到信号量大于等于一时。Release(释放)实际上是在信号量上执行加操作,对应于车辆离开停车场,该操作之所以叫做“释放”是因为释放了由信号量守护的资源。
  
  在iOS里我们使用GCD的dispatch_semaphore作为信号量的管理,非常方便。吃个栗子补补脑:
  *停车场剩余4个车位,那么即使同时来了四辆车也能停的下。如果此时来了五辆车,那么就有一辆需要等待。信号量的值就相当于剩余车位的数目,dispatch_semaphore_wait函数就相当于来了一辆车,dispatch_semaphore_signal就相当于走了一辆车。
停车位的剩余数目在创建初始化的时候由dispatch_semaphore_create(long value)指明了;调用一次dispatch_semaphore_signal,剩余的车位就增加一个;调用一次dispatch_semaphore_wait剩余车位就减少一个;
当剩余车位为0时,再来车(即调用dispatch_semaphore_wait)就只能等待。有可能同时有几辆车等待一个停车位。有些车主没有耐心,给自己设定了一段等待时间(dispatch_time_t),这段时间内等不到停车位就走了,如果等到了就开进去停车。而有些车主就想把车停在这,所以就一直等下去(DISPATCH_TIME_FOREVER)。*

dispatch_semaphore在iOS编程里的“应用”

在大多数情况下,做App开发,使用AFN异步请求,一个页面数据对应一个接口请求数据,很少涉及信号量的使用。如果一个页面需要多个请求,并且这几个请求并是无序的(如果一个请求需要依赖另一个请求的数据,我们只能称之为有序,一个页面需要两个以上的有序请求,会导致一些请求错误用户无法理解的情况,一般合并接口解决),这个时候我们可以有两个解决方案:

解决办法一

设一个全局变量,每次请求成功后该变量都+1,并且都检查该变量的值是不是3。如果是的话就刷新页面。伪代码如下:

int temp = 0;request A {   success {       temp++;       [self checkTemp];   }}request B { success {       temp++;       [self checkTemp];   }}request C { success {       temp++;       [self checkTemp];   }}checkTemp {        if (temp == 3){              refreshUI        }}

这是最简单最快解决问题方法,但是我认为是比较”笨”方法,这样的代码可读性会比较差。

解决办法二

使用信号量也同样可以解决这样的问题,并且不会造成代码可读性受损。直接上代码:

dispatch_group_t group = dispatch_group_create();dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{                [self requestA];            });            dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{                [self requestB];            });            dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{                [self requestC];            });dispatch_group_notify(group, dispatch_get_main_queue(), ^{                //刷新界面            });- (void)requestA{    dispatch_semaphore_t  sema = dispatch_semaphore_create(0);    [Request postWithURL:url params:params success:^(id response){        dispatch_semaphore_signal(sema);        //处理response    } failure:^(NSError *error) {        dispatch_semaphore_signal(sema);       //处理错误    }];    dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);}

requestB和requestC同上。我这就不写了。

还有一种情况就是,如果最后一个网络请求是依赖前面的所以请求
这里需要这样改一下

dispatch_group_notify(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{          //刷新界面});

上面requestA、requestB、requestC加了信号量同时使用GCD多线程的调度组后,他们也是异步执行,执行的先后顺不会卡住主线程。当A、B、Crequest的信号量全部都释放后,就会通知group_notify并执行其操作。
希望我的解决办法能帮助到有这些相关需求的朋友。

以上就是iOS开发之使用信号量实现AFN请求同步的全文介绍,希望对您学习和使用ios应用开发有所帮助.
  

原创粉丝点击