我们在处理一系列线程的时候,当数量达到一定量,在以前我们可能会选择使用NSOperationQueue来处理并发控制,但如何在GCD中快速的控制并发呢?答案就是dispatch_semaphore
信号量是一个整形值并且具有一个初始计数值,并且支持两个操作:信号通知和等待。当一个信号量被信号通知,其计数会被增加。当一个线程在一个信号量上等待时,线程会被阻塞(如果有必要的话),直至计数器大于零,然后线程会减少这个计数。
在GCD中有三个函数是semaphore的操作,分别是:
方法 | 执行目标 |
---|---|
dispatch_semaphore_create | 创建一个semaphore |
dispatch_semaphore_signal | 发送一个信号 |
dispatch_semaphore_wait | 等待信号 |
三个函数介绍: 第一个函数:创建一个整形数值的信号,即:信号的总量 第二个函数:发送一个信号,让信号总量增加1 第三个函数:首先判断信号量是否大于零,如果大于零则减掉1个信号量,往下执行,如果等于零则阻塞该线程
代码示例如下:
dispatch_semaphore_t signal = dispatch_semaphore_create(1);//创建一个为1信号量的信号
__block long x = 0;
NSLog(@"0_x:%ld",x);
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
sleep(1);
NSLog(@"waiting");
x = dispatch_semaphore_signal(signal);//此时信号量为0 对signal增加1 信号量变为1,
NSLog(@"1_x:%ld",x);
sleep(2);//睡两秒
NSLog(@"waking");
x = dispatch_semaphore_signal(signal);
NSLog(@"2_x:%ld",x);
});
// dispatch_time_t duration = dispatch_time(DISPATCH_TIME_NOW, 1*1000*1000*1000); //超时1秒
// dispatch_semaphore_wait(signal, duration);
x = dispatch_semaphore_wait(signal, DISPATCH_TIME_FOREVER);//此时信号量为1 所以执行下边,切对signal减掉1.然后信号量为0
NSLog(@"3_x:%ld",x);
x = dispatch_semaphore_wait(signal, DISPATCH_TIME_FOREVER);///此时信号量为0,永远等待,在等待的时候执行block了,在等待block时候block内对信号量增加了1,然后开始执行下边,并且信号量再次减掉1 变为0
NSLog(@"wait 2");
NSLog(@"4_x:%ld",x);
x = dispatch_semaphore_wait(signal, DISPATCH_TIME_FOREVER);//此时信号量为0,永远等待,在等待的时候执行block了,在等待block时候block内对信号量增加了1,然后开始执行下边,并且信号量再次减掉1 变为0
NSLog(@"wait 3");
NSLog(@"5_x:%ld",x);
代码执行结果:
// [84650:1805821] 0_x:0
// [84650:1805821] 3_x:0
// [84650:1805912] waiting
// [84650:1805912] 1_x:1
// [84650:1805821] wait 2
// [84650:1805821] 4_x:0
// [84650:1805912] waking
// [84650:1805912] 2_x:1
// [84650:1805821] wait 3
// [84650:1805821] 5_x:0
原文链接:http://www.jianshu.com/p/17afd8db5b79