当我们在处理多线程的时候,如果想控制并发线程的数量,我们会使用NSOperationQueue
的maxConcurrentOperationCount
来进行控制,所以遇到此类问题,我们一般会使用NSOperation+ NSOperationQueue来解决。
我们也可以使用GCD来解决这个问题,就是配合dispatch_semaphore来使用。
dispatch_semaphore就是信号量,在以前的Linux开发中就已经用过。信号量是一个整形值,在初始化的时候分配一个初始值,支持两个操作信号通知
和等待
。
等待
// 返回0:表示正常。返回非0:表示等待时间超时
long dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatch_time_t timeout);
信号通知
long dispatch_semaphore_signal(dispatch_semaphore_t dsema);
创建信号量
// 值得注意的是,这里的传入的参数value必须大于或等于0,否则dispatch_semaphore_create会返回NULL
dispatch_semaphore_t dispatch_semaphore_create(long value);
- 当一个信号量被通知
dispatch_semaphore_signal
,计数会加1;- 如果一个线程等待一个信号量
dispatch_semaphore_wait
,线程会被阻塞,直到计数器>0,此时开始运行,并且对信号量减1。
这样我们就可以根据 初始值 ,来控制可以有多少个并发的线程在运行。关于信号量,可以用停车位来比喻,如果停车场有5个停车位,都停满了,如果此时来了第6辆车,就需要等待,信号量的值就相当于剩余的车位的数量
。dispatch_semaphore_wait函数就相当于来了一辆车,dispatch_semaphore_signal就相当于走了一辆车。
dispatch_semaphore_wait中的参数timeout表示超时时间,如果等待期间没有获取到信号量或者信号量的值一直为0,那么等到timeout时,其所处线程自动执行其后语句。可取值为:DISPATCH_TIME_NOW
和 DISPATCH_TIME_FOREVER
,我们也可以自己设置一个dispatch_time_t的时间值,表示超时时间为这个时间之后。
- DISPATCH_TIME_NOW:超时时间为0,表示忽略信号量,直接运行
- ** DISPATCH_TIME_FOREVER**:超时时间为永远,表示会一直等待信号量为正数,才会继续运行
来看一个具体的例子:
信号量的初始值设置为:1,即最多只能又一个线程在run,可以验证一下运行结果,除去最开始的三个,后面的是每三秒打印一个,并且保证运行的顺序按照添加的顺序。
如果把创建信号量的值设置为4,即最多可以有4个线程同时运行,来看一下运行结果,是每三秒同时打印四个,且顺序不能保证。