浅谈iOS中的多线程-GCD

基本概念

1、进程与线程的关系?
进程有自己的内存空间,线程是执行进程的单元。所以,一个进程至少有一个线程
2、任务
就是block里面的代码块
3、队列
队列有FIFO的特性,多线程常见的队列分串行队列(serial)和并发队列(concurrent)。我们会往队列里面添加任务,他们的执行时间分别如下图


队列执行时间.jpeg

上图可以看出,遵循了FIFO原则,不管哪种队列,都是先放进的任务先执行。串行队列会等待前一个任务执行完再执行下一个任务,而并发队列则不会等待,只要上一个任务开始执行,不管是否执行完,下一个任务就开始了。

4、同步和异步
异步的特点:不会阻塞当前线程,具备开辟新线程的能力,但不一定每个异步任务都一定会开辟新线程执行
同步的特点:就在当前线程执行,会阻塞当前线程

GCD常用的API

1、同步串行队列

- (void)syncSerial{
    NSLog(@"begin");
    dispatch_queue_t queue = dispatch_queue_create("com.gcd", DISPATCH_QUEUE_SERIAL);
    dispatch_sync(queue, ^{
        NSLog(@"任务1");
    });
    dispatch_sync(queue, ^{
        NSLog(@"任务2");
    });
    dispatch_sync(queue, ^{
        NSLog(@"任务3");
    });
    NSLog(@"end");
}

打印顺序是: begin 任务1 任务2 任务3 end
特点:主线程顺序执行

2、异步串行队列

- (void)asyncSerial{
    NSLog(@"begin");
    dispatch_queue_t queue = dispatch_queue_create("com.gcd", DISPATCH_QUEUE_SERIAL);
    dispatch_async(queue, ^{//block1
        sleep(2);
        NSLog(@"1--%@",[NSThread currentThread]);
    });
    dispatch_async(queue, ^{//block2
        NSLog(@"2--%@",[NSThread currentThread]);
    });
    dispatch_async(queue, ^{//block3
        NSLog(@"3--%@",[NSThread currentThread]);
    });
    NSLog(@"end");
}

打印顺序是: begin end 1 2 3
特点:开辟了子线程,1 2 3顺序执行

3、异步并发

- (void)asyncConcurrent{
    NSLog(@"begin");
    dispatch_queue_t queue = dispatch_queue_create("com.gcd", DISPATCH_QUEUE_CONCURRENT);
    dispatch_async(queue, ^{//block1
        NSLog(@"1--%@",[NSThread currentThread]);
    });
    dispatch_async(queue, ^{//block2
        NSLog(@"2--%@",[NSThread currentThread]);
    });
    dispatch_async(queue, ^{//block3
        NSLog(@"3--%@",[NSThread currentThread]);
    });
    NSLog(@"end");
}

打印顺序是: begin end ( 1 2 3 无序)
特点:开辟了多个线程,但顺序不可控。

并发队列的异步任务一定会开辟新线程吗?
不一定

4、同步并发

- (void)syncConcurrent{
    NSLog(@"begin");
    dispatch_queue_t queue = dispatch_queue_create("com.gcd", DISPATCH_QUEUE_CONCURRENT);
    dispatch_sync(queue, ^{//block1
        NSLog(@"1--%@",[NSThread currentThread]);
    });
    dispatch_sync(queue, ^{//block2
        NSLog(@"2--%@",[NSThread currentThread]);
    });
    dispatch_sync(queue, ^{//block3
        NSLog(@"3--%@",[NSThread currentThread]);
    });
    NSLog(@"end");
}

打印顺序是: begin end 1 2 3 (和同步串行一样)
特点:当前线程执行

5、同步并发嵌套异步任务

- (void)syncConcurrentasync{
    NSLog(@"begin");
    dispatch_queue_t queue = dispatch_queue_create("com.gcd", DISPATCH_QUEUE_CONCURRENT);
    dispatch_sync(queue, ^{//block1
        NSLog(@"1");
        dispatch_async(queue, ^{//block2
            NSLog(@"2");
        });
        sleep(1);
        NSLog(@"3");
    });
    NSLog(@"end");
}

打印顺序是: begin 1 (2 3 顺序不确定) end

6、同步并发嵌套同步任务

- (void)syncConcurrentsync{
    NSLog(@"begin");
    dispatch_queue_t queue = dispatch_queue_create("com.gcd", DISPATCH_QUEUE_CONCURRENT);
    dispatch_sync(queue, ^{//block1
        NSLog(@"1");
        dispatch_sync(queue, ^{//block2
            sleep(2);
            NSLog(@"2");
        });
        NSLog(@"3");
    });
    NSLog(@"end");
}

打印顺序是: begin 1 2 3 end
特点:都在主线程,顺序执行(个人感觉没什么意义)

7、同步串行嵌套异步任务

- (void)syncSerialasync{
    NSLog(@"begin");
    dispatch_queue_t queue = dispatch_queue_create("com.gcd", DISPATCH_QUEUE_SERIAL);
    dispatch_sync(queue, ^{//block1
        NSLog(@"1");
        dispatch_async(queue, ^{//block2
            NSLog(@"2");
        });
        sleep(2);
        NSLog(@"3");
    });
    sleep(5);
    NSLog(@"end");
}

打印顺序是:begin 1 3 2 end
特点:串行队列有两个任务分别是block1 和 block2 ,block1 在主线程,block2 在子线程。串行队列,所以block1 执行完才会执行block2,所以2 在 3之后执行

8、同步串行嵌套同步任务死锁

- (void)syncSerialsync{
    NSLog(@"begin");
    dispatch_queue_t queue = dispatch_queue_create("com.gcd", DISPATCH_QUEUE_SERIAL);
    // 死锁,block1和block2相互等待
    dispatch_sync(queue, ^{//block1
        NSLog(@"1");
        dispatch_sync(queue, ^{//block2
            NSLog(@"2");
        });
        NSLog(@"3");
    });
    NSLog(@"end");
}

打印顺序是:begin 1 然后死锁
特点:死锁。因为block1在执行完1之后,在等block2执行,block2又在等block1执行完,所以就造成了死锁。

死锁:两个任务相互等待

9、主队列异步

- (void)main{
    NSLog(@"begin");
    dispatch_queue_t mainq = dispatch_get_main_queue();
    dispatch_async(mainq, ^{//block
        NSLog(@"1");
    });
    sleep(2);
    NSLog(@"end");
}

打印顺序是:begin end 1
特点:不开辟新线程,又不阻塞当前线程

主队列中异步:不会开辟新线程,不会阻塞当前线程
主队列中添加的任务需要等主线中的任务执行,再执行

10、主队列同步死锁

- (void)main{
    NSLog(@"begin");
    dispatch_queue_t mainq = dispatch_get_main_queue();
    dispatch_sync(mainq, ^{//block
        NSLog(@"1");
    });
}

打印顺序是:begin 然后死锁了
特点:死锁,因为main在等block执行,block又在等main执行完。

GCD中的高阶函数

开辟子线程是需要占内存和消耗cpu的资源的,main(1M)子线程(512kb)并不是线程越多越好,3-6条
1、dispatch_apply
开辟了子线程,index的打印、index和end的打印时无续的

- (void)apply{
    dispatch_queue_t queue = dispatch_queue_create("com.queue", DISPATCH_QUEUE_CONCURRENT);
    
//    for (int i = 0 ; i < 10; i++) {
//        dispatch_async(queue, ^{
//            NSLog(@"%@",[NSThread currentThread]);
//        });
//    }
    
    // 和上面加一个for循环一样的效果
    dispatch_apply(10, queue, ^(size_t index) {
        dispatch_async(queue, ^{
            NSLog(@"%zd",index);
        });
    });
    NSLog(@"end");
}

2、线程组
1 2 3都执行完才会执行end
group的任务执行完了,dispatch_group_notify才会执行

- (void)group{
    dispatch_group_t group = dispatch_group_create();
    dispatch_queue_t globalQ = dispatch_get_global_queue(0, 0);
    dispatch_group_async(group, globalQ, ^{
        dispatch_async(globalQ, ^{
            NSLog(@"1");;
        });
    });
    dispatch_group_async(group, globalQ, ^{
        dispatch_async(globalQ, ^{
            NSLog(@"2");
        });
    });
    dispatch_group_async(group, globalQ, ^{
        dispatch_async(globalQ, ^{
            NSLog(@"3");
        });
    });
    dispatch_group_notify(group, globalQ, ^{
        NSLog(@"end");
    });
}

3、队列优先级
一般情况下,不建议修改优先级,因为修改优先级也不会有很大的不同,反而会造成很多问题。
全局队列修改优先级,可以直接修改

dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0)

其他队列修改优先级,麻烦点,

dispatch_set_target_queue(q3, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0));

4、信号量
控制线程的并发数

dispatch_semaphore_create(1),这里为1时,就可以当锁用。

- (void)semaphore{
//    dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
//        for (int i = 0 ; i < 10; i++) {
//            dispatch_async(queue, ^{
//                NSLog(@"%@",[NSThread currentThread]);
//            });
//        }
    
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(2);
    dispatch_apply(10, dispatch_get_global_queue(0, 0), ^(size_t index) {
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
        NSLog(@"%zd",index);
        sleep(1);
        NSLog(@"%zd ok",index);
        dispatch_semaphore_signal(semaphore);
    });
}

常见用法是:dispatch_semaphore_create(0),当这里的0变为小于0,就取消等待了,如下

        dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
        __block AVAsset *avAsset = nil;
        [[PHImageManager defaultManager] requestAVAssetForVideo:asset options:nil resultHandler:^(AVAsset * _Nullable aVasset, AVAudioMix * _Nullable audioMix, NSDictionary * _Nullable info) {
            AVURLAsset *urlAsset = (AVURLAsset *)aVasset;
            avAsset = urlAsset;
            dispatch_semaphore_signal(semaphore);
        }];
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);

5、栅栏函数(可以用来实现NSOperation中的依赖

dispatch_barrier_async的queue不能是dispatch_get_global_queue(0, 0),只能是自己create的,不然dispatch_barrier_async就没用了

可以保证 dispatch_barrier_async 前面的任务在barrier前面执行,后面的任务在barrier后面执行。

- (void)barrier{
    dispatch_queue_t queue = dispatch_queue_create(NULL, DISPATCH_QUEUE_CONCURRENT);
    dispatch_async(queue, ^{
        sleep(2);
        NSLog(@"1");
    });
    dispatch_async(queue, ^{
        NSLog(@"2");
    });
    dispatch_barrier_async(queue, ^{
        NSLog(@"barrier");
    });
    dispatch_async(queue, ^{
        sleep(2);
        NSLog(@"3");
    });
    dispatch_async(queue, ^{
        NSLog(@"4");
    });
}

线程安全问题

1、何为线程安全,就是读取内存数据,结果是可预见的,那就是线程安全的

如何解决线程安全问题?
加锁。iOS开发常用的锁有:用信号量、NSLock、@synchronized

2、有哪些锁?
如图


lock.jpeg

synchronized锁的用法:

- (void)lock{
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        @synchronized(self){
            NSLog(@"1");
            sleep(2);
            NSLog(@"1 ok");
        }
        
    });
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        @synchronized(self){
            NSLog(@"2");
            sleep(2);
            NSLog(@"2 ok");
        }
    });
}

3、atomic:原子特性,set/get线程安全

为什么开发中不使用atomic?
1、并不是真正意义的线程安全
2、UIKit的类就不用atomic,因为这个框架里的属性都在主线程执行,本身就是线程安全的
3、耗性能

为了剖析原理,我们来分析两个demo
示例1:

@property (nonatomic, strong) NSString *target;
- (void)test{
    dispatch_queue_t queue = dispatch_queue_create(NULL, DISPATCH_QUEUE_CONCURRENT);
    for (int i = 0; i < 10000; i++) {
        dispatch_async(queue, ^{
            self.target = [NSString stringWithFormat:@"hello-%d",i];
        });
    }
}

执行这个方法会报EXC_BAD_ACCESS错误(向一个已释放的对象发送消息)。因为,target是非原子性的,改为atomic就没事。
虽然我们现在是ARC,其实在调用target的setter方法时,会做一次release和retain,因为该任务异步的,所以某些时候release都执行两次了。

- (void)setTarget:(NSString *)target{
 if (_target != target) {
     [_target release];
     [target retain];
     _target = target;
    }
 }

你以为atomic就一定安全吗?请看示例2

@property (atomic, assign) int number;
- (void)test2{
    _number = 0;
    dispatch_apply(10000, dispatch_get_global_queue(0, 0), ^(size_t index) {
        self.number++;
    });
    NSLog(@"total = = %d",self.number);
}

理想状态会答应1000,其实不然。不信试试?

number++ 相当于做了下面操作,所以并不安全
int temp = number + 1
num = temp

那我们加个锁试试

- (void)test2{
    _number = 0;
    NSLock *lock = [[NSLock alloc] init];
    dispatch_apply(10000, dispatch_get_global_queue(0, 0), ^(size_t index) {
        [lock lock];
        self.number++;
        [lock unlock];
    });
    NSLog(@"total = = %d",self.number);
}

对啦,1000打印出来了。

所以,atomic针对简单的set/get是安全的,遇到复合操作就不是安全的了。

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

推荐阅读更多精彩内容

  • 一、提问 1、如何叙事? 2、叙事疗法有什么特别的原因有好的效果? 二、预习 在预习的阶段,就是看到了这本书的结构...
    吾宗老孙子阅读 433评论 0 1
  • 作为专业性文章,舍弃了观赏性,只是运用思维导图的方式将EPC质量控制的各项内容清晰列出。
    Blockchain007阅读 214评论 0 1
  • 潘蔚~常州新日催化剂有限公司 【日精进打卡第七十七天】 【知~学习】 《六项精进》1遍共104遍 《大学》1遍共1...
    BOOpan阅读 149评论 0 0
  • 初识林丽是在2016年,她是黑龙江人,和父母一起在坡月养病。那个时候的她,情绪低落,脸色也不好,又黄又瘦。一看,就...
    云淑阅读 1,071评论 1 7
  • 一直以来,我喜欢跟孩子们相处,并不是害怕复杂的社会,而是孩子的童真会带给我们很多很多的欢乐。 我们的心简单,社会也...
    井溢阅读 435评论 0 2