GCD 线程安全同步-信号量

GCD 线程安全同步

学习、记录与分享

GCD 与 NSThread比较

  • GCD会自动利用更多的CPU内核、 会自动管理线程的生命周期 (创建线程、调度线程、销毁线程)开发者只需用GCD函数创建任务,加入队列,GCD会根据CPU内核自动创建线程(创建多少,怎么创建不许要管)去完成任务。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
   }); 

   ````
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,324评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,356评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,328评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,147评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,160评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,115评论 1 296
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,025评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,867评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,307评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,528评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,688评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,409评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,001评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,657评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,811评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,685评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,573评论 2 353

推荐阅读更多精彩内容