信号量(Semaphore),有时被称为信号灯,是在多线程环境下使用的一种设施,是可以用来保证两个或多个关键代码段不被并发调用。在进入一个关键代码段之前,线程必须获取一个信号量;一旦该关键代码段完成了,那么该线程必须释放信号量。其它想进入该关键代码段的线程必须等待直到第一个线程释放信号量。为了完成这个过程,需要创建一个信号量VI,然后将Acquire Semaphore VI以及Release Semaphore VI分别放置在每个关键代码段的首末端。确认这些信号量VI引用的是初始创建的信号
以一个停车场的运作为例。简单起见,假设停车场只有三个车位,一开始三个车位都是空的。这时如果同时来了五辆车,看门人允许其中三辆直接进入,然后放下车拦,剩下的车则必须在入口等待,此后来的车也都不得不在入口处等待。这时,有一辆车离开停车场,看门人得知后,打开车拦,放入外面的一辆进去,如果又离开两辆,则又可以放入两辆,如此往复。
在这个停车场系统中,车位是公共资源,每辆车好比一个线程,看门人起的就是信号量的作用。
抽象的来讲,信号量的特性如下:信号量是一个非负整数(车位数),所有通过它的线程/进程(车辆)都会将该整数减一(通过它当然是为了使用资源),当该整数值为零时,所有试图通过它的线程都将处于等待状态。在信号量上我们定义两种操作: Wait(等待) 和 Release(释放)。当一个线程调用Wait操作时,它要么得到资源然后将信号量减一,要么一直等下去(指放入阻塞队列),直到信号量大于等于一时。Release(释放)实际上是在信号量上执行加操作,对应于车辆离开停车场,该操作之所以叫做“释放”是因为释放了由信号量守护的资源。
以上是百度百科内容方便理解信号量
对信号量有4种操作:
1. 初始化(initialize),也叫做建立(create):
dispatch_semaphore_t dispatch_semaphore_create( long value);
2. 等信号(wait),也可叫做挂起(suspend):
线程等待信号量,如果值大于0,则获得,值减一;如果值等于0,则线程进入睡眠状态,只到信号量值大于0或者超时。
long dispatch_semaphore_wait( dispatch_semaphore_t dsema, dispatch_time_t timeout);
如果返回的结果是0说明线程线程被成功唤醒,非0说明超时;
3. 给信号(signal)或发信号(post):
执行释放信号量,则值加一;如果此时有正在等待的线程,则唤醒该线程。
long dispatch_semaphore_signal( dispatch_semaphore_t dsema);
如果返回的是非零的数字则说明有一个线程被唤醒,返回0则说明没有线程被唤醒(说明没有wait的线程)
4.清理(destroy):
dispatch_release(object) //在ARC中不需要此步骤
示例代码
dispatch_group_t group = dispatch_group_create();
dispatch_semaphore_t semaphore = dispatch_semaphore_create(2);
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
for (int i = 0; i < 100; i++)
{
long l1 = dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"l1:%ld",l1);
NSLog(@"semaphore:%@",semaphore.accessibilityValue);
dispatch_group_async(group, queue, ^{
NSLog(@"%i",i);
sleep(2);
long l2 = dispatch_semaphore_signal(semaphore);
NSLog(@"l2:%ld",l2);
});
}
dispatch_group_wait(group, DISPATCH_TIME_FOREVER);