dispatch_semaphore 的用法(线程加锁)

转自:综合

dispatch_semaphore 信号量基于计数器的一种多线程同步机制。在多个线程访问共有资源时候,会因为多线程的特性而引发数据出错的问题。

复制代码

    dispatch_queue_t queue = dispatch_get_global_queue(00);

    dispatch_semaphore_t semaphore = dispatch_semaphore_create(1);

    NSMutableArray *array = [NSMutableArrayarray];

    for (int index = 0; index < 100000; index++) {

        dispatch_async(queue, ^(){

            dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);//

            NSLog(@"addd :%d", index);

            [array addObject:[NSNumber numberWithInt:index]];

            dispatch_semaphore_signal(semaphore);

        });

    }

复制代码

dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER); 如果semaphore计数大于等于1.计数-1,返回,程序继续运行。如果计数为0,则等待。这里设置的等待时间是一直等待。dispatch_semaphore_signal(semaphore);计数+1.在这两句代码中间的执行代码,每次只会允许一个线程进入,这样就有效的保证了在多线程环境下,只能有一个线程进入。



当我们在处理一系列线程的时候,当数量达到一定量,在以前我们可能会选择使用NSOperationQueue来处理并发控制,但如何在GCD中快速的控制并发呢?答案就是dispatch_semaphore,对经常做unix开发的人来讲,我所介绍的内容可能就显得非常入门级了,信号量在他们的多线程开发中再平常不过了。
  信号量是一个整形值并且具有一个初始计数值,并且支持两个操作:信号通知和等待。当一个信号量被信号通知,其计数会被增加。当一个线程在一个信号量上等待时,线程会被阻塞(如果有必要的话),直至计数器大于零,然后线程会减少这个计数。
  在GCD中有三个函数是semaphore的操作,分别是:
  dispatch_semaphore_create   创建一个semaphore
  dispatch_semaphore_signal   发送一个信号
  dispatch_semaphore_wait    等待信号
  简单的介绍一下这三个函数,第一个函数有一个整形的参数,我们可以理解为信号的总量,dispatch_semaphore_signal是发送一个信号,自然会让信号总量加1,dispatch_semaphore_wait等待信号,当信号总量少于0的时候就会一直等待,否则就可以正常的执行,并让信号总量-1,根据这样的原理,我们便可以快速的创建一个并发控制来同步任务和有限资源访问控制。

[cpp]  view plain copy
  1. dispatch_group_t group = dispatch_group_create();   
  2.     dispatch_semaphore_t semaphore = dispatch_semaphore_create(10);   
  3.     dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);   
  4.     for (int i = 0; i < 100; i++)   
  5.     {   
  6.         dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);   
  7.         dispatch_group_async(group, queue, ^{   
  8.             NSLog(@"%i",i);   
  9.             sleep(2);   
  10.             dispatch_semaphore_signal(semaphore);   
  11.         });   
  12.     }   
  13.     dispatch_group_wait(group, DISPATCH_TIME_FOREVER);   
  14.     dispatch_release(group);   
  15.     dispatch_release(semaphore);   

  简单的介绍一下这一段代码,创建了一个初使值为10的semaphore,每一次for循环都会创建一个新的线程,线程结束的时候会发送一个信号,线程创建之前会信号等待,所以当同时创建了10个线程之后,for循环就会阻塞,等待有线程结束之后会增加一个信号才继续执行,如此就形成了对并发的控制,如上就是一个并发数为10的一个线程队列。


简单示例

 

[cpp]  view plain copy
  1. __block BOOL isok = NO;  
  2.       
  3.     dispatch_semaphore_t sema = dispatch_semaphore_create(0);  
  4.     Engine *engine = [[Engine alloc] init];  
  5.     [engine queryCompletion:^(BOOL isOpen) {  
  6.         isok = isOpen;  
  7.         dispatch_semaphore_signal(sema);  
  8.     } onError:^(int errorCode, NSString *errorMessage) {  
  9.         isok = NO;  
  10.         dispatch_semaphore_signal(sema);  
  11.     }];  
  12.       
  13.     dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);  
  14.     dispatch_release(sema);  

你可能感兴趣的:(多线程,线程,并发,Semaphore,gcd)