iOS多线程之GCD

Grand Central Dispatch (GCD)是Apple开发的一个多核编程的解决方法。dispatch queue分成以下三种:

  • 运行在主线程的Main_queue,通过dispatch_get_main_queue获取。dispatch_get_main_queue也是一种dispatch_queue_t
  • 并行队列global dispatch queue,通过dispatch_get_global_queue获取,由系统创建三个不同优先级的dispatch_queue。并行队列的执行顺序与其加入队列的顺序相同。
  • 串行队列serial queues一般用于按顺序同步访问,可创建任意数量的串行队列,各个串行队列之间是并发的。
    当想要任务按照某一个特定的顺序执行时,串行队列是很有用的。串行队列在同一个时间只执行一个任务。我们可以使用串行队列代替锁去保护共享的数据。和锁不同,一个串行队列可以保证任务在一个可预知的顺序下执行。
    serial queues通过dispatch_queue_create创建,可以使用函数dispatch_retaindispatch_release去增加或者减少引用计数。
严格来说,GCD并非多线程,而是一个队列。

GCD的用法:

1、并行队列 + 同步执行
- (void)gcdDemo1 {
    /* 
        DISPATCH_QUEUE_CONCURRENT 并发
        DISPATCH_QUEUE_SERIAL 串行
     */
    
    NSLog(@"嘿嘿-----------");
    dispatch_queue_t q = dispatch_queue_create("ptl", DISPATCH_QUEUE_CONCURRENT);
    dispatch_sync(q, ^{
        for (NSInteger i = 0; i < 10; i ++) {
            // 任务在并发子线程执行 所以下面的"哈哈" 可能在for之前 也可能之后
            dispatch_async(q, ^{
                NSLog(@"😆(%zd)--%@", i, [NSThread currentThread]);
            });
        }
    });
    
    NSLog(@"哈哈-----------");
}
2、 并行队列 + 异步执行
- (void)gcdDemo2 {
    /*
     先执行"嘿嘿", 下面的代码都是随机执行 "哈哈"可能在for前,后或者中间打印
     
     */
    NSLog(@"嘿嘿-----------");
    
    dispatch_queue_t q = dispatch_queue_create("ptl", DISPATCH_QUEUE_CONCURRENT);
    dispatch_async(q, ^{
        for (NSInteger i = 0; i < 10; i ++) {
            dispatch_async(q, ^{
                NSLog(@"😆(%zd)--%@", i, [NSThread currentThread]);
            });
        }
    });
    NSLog(@"哈哈-----------");
}
3、串行队列 + 同步执行
- (void)gcdDemo3 {
    
    /*
     DISPATCH_QUEUE_SERIAL : 串行
     按照顺序执行
     */
    NSLog(@"嘿嘿-----------");
    dispatch_queue_t q = dispatch_queue_create("ptl", DISPATCH_QUEUE_SERIAL);
    dispatch_sync(q, ^{
        for (NSInteger i = 0; i < 10; i ++) {
            dispatch_async(q, ^{
                NSLog(@"😆(%zd)--%@", i, [NSThread currentThread]);
            });
        }
    });
    NSLog(@"哈哈-----------");
}
4、串行队列 + 异步执行

- (void)gcdDemo4 {
    
    /*
        先执行"嘿嘿", 后面的会随机执行, "哈哈"可能会插入到for中打印
        for是在串行异步执行 所以子线程会执行完当前任务后才会执行下次任务
     */
    
    NSLog(@"嘿嘿-----------");
    dispatch_queue_t q = dispatch_queue_create("ptl", DISPATCH_QUEUE_SERIAL);
    dispatch_async(q, ^{
        for (NSInteger i = 0; i < 2; i ++) {
            dispatch_async(q, ^{
                NSLog(@"😆(%zd)--%@", i, [NSThread currentThread]);
            });
        }
    });
    NSLog(@"哈哈-----------");
    
}
5、主队列 + 同步执行
- (void)gcdDemo5 {
    
    /*
        "嘿嘿"等待for的任务执行完往下执行 而for又等待"嘿嘿"执行完往下执行 所以互相等待 程序卡死 崩溃
     */
    NSLog(@"嘿嘿-----------");
    dispatch_sync(dispatch_get_main_queue(), ^{
        for (NSInteger i = 0; i < 10; i ++) {
            NSLog(@"😆(%zd)--%@", i, [NSThread currentThread]);
        }
    });
    
    NSLog(@"哈哈-----------");
}
6、主队列 + 异步执行
- (void)gcdDemo6 {
    /*
        先执行"嘿嘿"->"哈哈"->for里面 for里面的任务会在主队列执行 添加到主队列后不会立马就执行的 会等待一会 所以"哈哈"在for之前就执行了 因为它本身就在主线程中无需等待
     */
    NSLog(@"嘿嘿-----------");
    dispatch_async(dispatch_get_main_queue(), ^{
        for (NSInteger i = 0; i < 10; i ++) {
            NSLog(@"😆(%zd)--%@", i, [NSThread currentThread]);
        }
    });
    
    NSLog(@"哈哈-----------");
}
7、栅栏方法
  • dispatch_barrier_async
    有时需要异步执行两组操作,而且第一组操作执行完之后,才能开始执行第二组操作。这样我们就需要一个相当于栅栏一样的一个方法将两组异步执行的操作组给分割起来,操作组里可以包含一个或多个任务。这就需要用到dispatch_barrier_async方法在两个操作组间形成栅栏。
- (void)gcdDemo7 {

    /*
        "1"和"2"处于子线程会随机打印顺序 然后打印栅栏"3" 最后打印"4"
     */
    
   dispatch_queue_t q = dispatch_queue_create("ptl", DISPATCH_QUEUE_CONCURRENT);
    dispatch_async(q, ^{
        NSLog(@"----1-----%@", [NSThread currentThread]);
    });
    
    dispatch_async(q, ^{
        NSLog(@"----2-----%@", [NSThread currentThread]);
    });
    
    dispatch_barrier_async(q, ^{
        for (NSInteger i = 0; i < 10; i ++) {
//            // 加入到异步后 下面的"4"可能在for之前打印, 可能在中间打印, 也可能在for后面打印
//            dispatch_async(q, ^{
//                NSLog(@"----3 barrier-----%@", [NSThread currentThread]);
//            });
            
            NSLog(@"----3 barrier-----%@", [NSThread currentThread]);
        }
    });
    
    dispatch_async(q, ^{
        NSLog(@"----4-----%@", [NSThread currentThread]);
    }); 
}
8、快速迭代方法 dispatch_apply

dispatch_apply函数是dispatch_sync函数和Dispatch_Group的关联API,该函数按指定的次数将指定的Block追加到指定的Dispatch_Queue中,并等到全部的处理执行结束。

- (void)gcdDemo8 {

    dispatch_queue_t q = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    /**
     同时遍历6次 不按顺序输出
     @param iterations 遍历的次数
     @param queue 在哪个队列
     @param size_t index下标号
     */
    dispatch_apply(6, q, ^(size_t index) {
        NSLog(@"%zd----%@", index, [NSThread currentThread]);
    });
}
9、队列组 dispatch_group

多个任务处理完毕之后想执行结束处理,这种需求会经常出现。如果只是使用一个Serial Dispatch Queue(串行队列)时,只要将想执行的处理全部追加到该串行队列中并在最后追加结束处理即可,但是在使用Concurrent Queue 时,可能会同时使用多个Dispatch Queue时,代码就会变得很复杂。在这种情况下,就可以使用Dispatch Group。

- (void)gcdDemo9 {
    /*
        "3"会等待前面的"1""2"打印完才会打印, "1""2"异步操作会交替打印
     */
    // 创建一个组队列
    dispatch_group_t group = dispatch_group_create();
    // 创建一个异步组队列
    dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        // 执行1个耗时的异步操作
        for (NSInteger i = 0; i < 10; i ++) {
            NSLog(@"1----%zd---%@",i, [NSThread currentThread]);
        }
    });
    
    dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        // 执行1个耗时的异步操作
        for (NSInteger i = 0; i < 10; i ++) {
            NSLog(@"2----%zd---%@",i, [NSThread currentThread]);
        }
    });
    
    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
        // 等前面的异步操作都执行完毕后,回到主线程...
        for (NSInteger i = 0; i < 10; i ++) {
            NSLog(@"3----%zd---%@",i, [NSThread currentThread]);
        }
    });
}
10、信号量dispatch_semaphore

项目中的业务接口请求的时候需要Token验证。我们最简化这个需求就是:两个请求,请求1成功返回所需参数之后,才能开始请求2。

// 信号量
- (void)dispatch_semaphore {
    
    /*
     //创建信号量
     dispatch_semaphore_create
     //发送信号量
     dispatch_semaphore_signal
     //等待信号量
     dispatch_semaphore_wait
     */
    
    dispatch_async(dispatch_queue_create("ptl", DISPATCH_QUEUE_CONCURRENT), ^{
        
        dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);

        // 请求接口1
        [self getToken:semaphore];
        //此时的信号量为0,只有token请求成功发送信号量之后,才会往下执行[self request]方法,否则会一直等下去;
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);

        // 请求接口2
        [self requestData1:semaphore];
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);

        // 请求接口3
        [self requestData2];
    });
}

- (void)getToken:(dispatch_semaphore_t)semaphore
{
    // 模仿网络请求
    for (NSInteger i = 0; i < 20; i ++) {
        NSLog(@"请求1-------%zd", i);
        // 模仿请求成功
        //成功拿到token,发送信号量
        dispatch_semaphore_signal(semaphore);
    }
    NSLog(@"------------请求1成功-----------------");
}

- (void)requestData1:(dispatch_semaphore_t)semaphore {
    for (NSInteger i = 0; i < 20; i ++) {
        NSLog(@"请求2-------%zd", i);
        dispatch_semaphore_signal(semaphore);
    }
    NSLog(@"------------请求2成功-----------------");
}

- (void)requestData2 {
    for (NSInteger i = 0; i < 20; i ++) {
        NSLog(@"请求3-------%zd", i);
    }
    NSLog(@"------------请求3成功-----------------");
}

Github: https://github.com/soliloquy-local/GCD.git
以上就是对GCD各个情况的简单介绍,主要以代码为主...😁

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

推荐阅读更多精彩内容