iOS干货汇总,让开发变得简单。iOS学习iOS开发

GCD 线程安全同步-信号量

2016-09-12  本文已影响124人  Yan青天

GCD 线程安全同步

学习、记录与分享

GCD 与 NSThread比较

GCD核心

队列

//DISPATCH_QUEUE_CONCURRENT代表并发队列
//DISPATCH_QUEUE_SERIAL或NULL代表串行队列
//标示符代表这个队列的一个标记
dispatch_queue_t qune = dispatch_queue_create("创建", DISPATCH_QUEUE_CONCURRENT);
//通过获取全局队列来获得一个并发队列
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
//这两个参数,第一个是优先级一般用默认,第二个直接设为0 是一个保留标记,实际作用不大

* 串行队列
+ 队列里的任务会以串行的形式一个一个按顺序执行
+ 创建串行队列



//直接创建
//DISPATCH_QUEUE_SERIAL或NULL代表串行队列
dispatch_queue_t queue = dispatch_queue_create("标示符", DISPATCH_QUEUE_SERIAL);
//获得主队列,也是一种串行队列
dispatch_queue_t queue = dispatch_get_main_queue();


#####函数
* 同步函数:执行之后不立即返回,等待任务完成才返回,会阻塞当下线程


//queue代表你要放入的队列
dispatch_sync(queue, ^{
//在这里写要执行的代码
});
`


* 异步函数:执行之后立即返回,不会阻塞当下线程

//queue代表你要放入的队列
dispatch_async(queue, ^{
//在这里写要执行的代码
});


* 栅栏函数

//隔断函数,前面执行完才会执行这个函数,这个函数执行完才会执行其他后面的函数
dispatch_barrier_async(dispatch_queue_t queue, dispatch_block_t block);

####不同的队列与函数的组合会有不同的效果
* 并行队列+异步函数:创建新的线程,并行执行任务
* 并行队列+同步函数:没有新的线程,串行执行任务
* 串行队列+异步函数:创建新的线程,串行执行任务
* 串行队列+同步函数:没有新的线程,串行执行任务
* 主队列+异步函数:没有新的线程,串行执行任务
* 主队列+同步函数:没有新的线程,串行执行任务(主队列虽然也是串行队列)

####下面通过GCD实现单一资源线程安全的多读单写
#####一 信号量#####
简单来说就是控制访问资源的数量,比如系统有两个资源可以被利用,同时有三个线程要访问,只能允许两个线程访问,第三个应当等待资源被释放后再访问。

#####二 使用#####
* 创建信号量

dispatch_semaphore_t semaphore = dispatch_semaphore_create(1);
// 1 是信号量的初始值 这里只允许一个线程访问

* 提高信号量

dispatch_semaphore_signal(semaphore)

* 等待降低信号量

dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
// 如果semaphore计数大于等于1.计数-1,返回,程序继续运行。如果计数为0,则等待。


测试

dispatch_queue_t qune = dispatch_queue_create("x", DISPATCH_QUEUE_CONCURRENT);
dispatch_semaphore_t semaphore = dispatch_semaphore_create(1);
__block int i = 1 , n = 1;
for (int index = 0; index < 100; index++) {

    dispatch_async(qune, ^(){
        i++;
        NSLog(@"%d %d\n", index,i);
        
    });
    
}

/*
 只能单写
 */
for (int index = 0; index < 100; index++) {
    
    dispatch_async(qune, ^(){
        
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);//
        n++;
        NSLog(@"%d xx  %d\n", index,n);

// sleep(1);
dispatch_semaphore_signal(semaphore);

    });
    
}

打印LOG:
2016-09-12 14:32:11.849 Mansory[4745:108806] 0 2
2016-09-12 14:32:11.849 Mansory[4745:108807] 2 4
2016-09-12 14:32:11.849 Mansory[4745:108805] 1 3
2016-09-12 14:32:11.850 Mansory[4745:108887] 3 5
2016-09-12 14:32:11.850 Mansory[4745:108888] 4 6
2016-09-12 14:32:11.851 Mansory[4745:108807] 6 8
2016-09-12 14:32:11.851 Mansory[4745:108806] 5 7
2016-09-12 14:32:11.851 Mansory[4745:108805] 7 9
2016-09-12 14:32:11.851 Mansory[4745:108887] 8 10
2016-09-12 14:32:11.851 Mansory[4745:108888] 9 11
.
.
index 是乱序的 因为是异步的   i值是乱序的 因为资源竞争的问题导致

2016-09-12 14:32:11.863 Mansory[4745:108910] 0 xx  2
2016-09-12 14:32:11.863 Mansory[4745:108807] 99 101
2016-09-12 14:32:11.868 Mansory[4745:108807] 39 xx  3
2016-09-12 14:32:11.868 Mansory[4745:108807] 42 xx  4
2016-09-12 14:32:11.868 Mansory[4745:108807] 43 xx  5
2016-09-12 14:32:11.868 Mansory[4745:108807] 45 xx  6
2016-09-12 14:32:11.868 Mansory[4745:108807] 46 xx  7
2016-09-12 14:32:11.868 Mansory[4745:108807] 48 xx  8
2016-09-12 14:32:11.869 Mansory[4745:108927] 50 xx  9
2016-09-12 14:32:11.869 Mansory[4745:108927] 51 xx  10
2016-09-12 14:32:11.869 Mansory[4745:108927] 53 xx  11
2016-09-12 14:32:11.869 Mansory[4745:108927] 54 xx  12
2016-09-12 14:32:11.869 Mansory[4745:108927] 56 xx  13
2016-09-12 14:32:11.869 Mansory[4745:108927] 57 xx  14
2016-09-12 14:32:11.870 Mansory[4745:108927] 59 xx  15
2016-09-12 14:32:11.870 Mansory[4745:108927] 61 xx  16
.
.
index 是乱序的 是异步的 n值是递增的有序的 资源安全

#### dispatch_group_t 队列同步
* 手动管理group关联的block的运行状态

dispatch_group_t group = dispatch_group_create();
__weak typeof(self) this = self;

dispatch_group_enter(group);
[this productEvlute]; //网络请求

self.requestProductSucess = ^{
//请求成功回调
     dispatch_group_leave(group);
};

dispatch_group_enter(group);
[this estimateServive]; //网络请求耗时操作
 
 self.requestServiceSucess = ^{
 //请求成功
      dispatch_group_leave(group);
  };
}
.
.

dispatch_group_notify(group, dispatch_get_main_queue(), ^{
//任务都完成回调
[this checkIFAllCommitSucess];
});
/*
进入dispatch_group_enter和退出dispatch_group_leave次数必须匹配
*/

*

   ````
   dispatch_group_async(group, queue, ^{ 
       //任务1
   });  
   dispatch_group_async(group, queue, ^{ 
       //任务2
   }); 
   .
   .
   dispatch_group_notify (group, queue, ^{ 
       //任务都complete
   }); 

   ````
上一篇 下一篇

猜你喜欢

热点阅读