GCD的简单使用

作者: 马戏团小丑 | 来源:发表于2016-05-29 21:22 被阅读394次

1> 队列的类型

  • 并发队列 多个任务并发(同时)执行
    a.自己创建 dispatch_queue_create 第一个参数:C语言的字符串,标签 第二个参数:创造哪种类型的队列 DISPATCH_QUEUE_CONCURRENT(并发)
    b.获得全局的并发队列: dispatch_get_global_queue 第一个参数:队列的优先级
  • 串行队列 一个任务执行完毕后,再执行下一个任务
    a.自己创建 dispatch_queue_create 第二个参数:创造哪种类型的队列 DISPATCH_QUEUE_SERIAL(串行)
    b.获得主队列 dispatch_get_main_queue

2> 执行任务的方法类型

同步(sync)执行 在当前线程中执行任务,不具备开启新线程的能力

dispatch_sync(dispatch_queue_tqueue, dispatch_block_tblock);

异步(async)执行 在新的线程中执行任务,具备开启新线程的能力

dispatch_async(dispatch_queue_tqueue, dispatch_block_tblock);

3> 队列和方法的配合使用


同步+主队列:注意在主线程中执行会发生死锁,在子线程中执行不会

4> GCD线程间通信

子线程回到主线程

子线程回到主线程
dispatch_async(
dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
   // 执行耗时的异步操作...
   dispatch_async(dispatch_get_main_queue(), ^{
       // 回到主线程,执行UI刷新操作
   });
});

5> 其他方法

  • 1.dispatch_once 程序运行中只执行一次
//使用dispatch_once函数能保证某段代码在程序运行过程中只被执行1次
static dispatch_once_t onceToken;dispatch_once(&onceToken, ^{ 
    // 只执行1次的代码(这里面默认是线程安全的
)});
  • 2.dispatch_after 延时执行
    iOS常见的延时执行有3种方式 (还有使用定时器NSTimer)
    a.调用NSObject的方法
[self performSelector:@selector(run) withObject:nil afterDelay:2.0];

b.使用GCD函数

dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ 
    //dispatch_get_main_queue可以换成并发队列,这样就在子线程执行
    // 2秒后异步执行这里的代码,在主线程执行...

});
  • 3.队列组(同栅栏函数)dispatch_group_async\dispatch_group_notify
    有这么1种需求: 首先:分别异步执行2个耗时的操作, 其次:等2个异步操作都执行完毕后,再回到主线程执行操作
    如果想要快速高效地实现上述需求,可以考虑用队列组
dispatch_group_tgroup group =  dispatch_group_create();
dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    // 执行1个耗时的异步操作
});
dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    // 执行1个耗时的异步操作
});
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
    // 等前面的异步操作都执行完毕后,回到主线程...
});

进入群组和离开群组(以前的方法)

{
    //0.创建队列组
    dispatch_group_t group = dispatch_group_create();
    //1.创建队列
    dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
    ////任务1
    //2.在该方法后面的任务会被纳入到队列组的监听范围中
    //dispatch_group_enter|dispatch_group_leave 配对使用
    dispatch_group_enter(group);
    dispatch_async(queue, ^{
        NSLog(@"download1---%@",[NSThread currentThread]);
    });
   dispatch_group_leave(group);

    //任务2
    dispatch_group_enter(group);
    dispatch_async(queue, ^{
        NSLog(@"download2---%@",[NSThread currentThread]);
        [NSThread sleepForTimeInterval:3.0];
    });
      dispatch_group_leave(group);

    //3.当队列组中所有的任务都执行完毕之后会通知group执行dispatch_group_notify方法
    //内部是异步执行的,不会阻塞
    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
        NSLog(@"队列组中所有的任务都执行完毕了");
    });
    //等到队列组中所有任务的执行,一直等待|阻塞
   // dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
    NSLog(@"----end---");
}
  • 4.栅栏函数(控制任务的执行顺序) 该函数前面必须执行完才能执行后面的
    在使用栅栏函数的时候,苹果官方明确规定栅栏函数只有在和使用create函数自己的创建的并发队列一起使用的时候才有效
dispatch_barrier_async(queue, ^{ 
    NSLog(@"--dispatch_barrier_async-");
});
  • 5.快速迭代(开多个线程并发完成迭代操作)
    会阻塞当前线程直到所有循环迭代执行完成
dispatch_apply(subpaths.count, queue, ^(size_t index) { 
    //参数分别是:1.要迭代的次数 2.队列,要传并发队列 3.block封装任务 
});

相关文章

  • GCD介绍

    一、GCD简单介绍 什么是GCD GCD优势 任务和队列 GCD有2个核心概念 GCD的使用就2个步骤 将任务添加...

  • 多线程和AFN网络框架配合使用

    ios的多线程一般有NSOperation和GCD.NSOperation基本使用: GCD基本使用: 简单的多线...

  • GCD简单使用

  • GCD简单使用

    开发中有时候可能会有一些很耗时的操作,不可能说让屏幕一直卡着,等操作完成,那体验就很尴尬了。所以多线程还是需要来一...

  • GCD简单使用

    一.普通使用 1.后台执行 2.主线程执行 3.一次性执行 4.延迟两秒执行 自定义使用 1.自定义队列 2.两个...

  • GCD简单使用

    基础使用 GCD的延时执行方法 dispatch_after GCD的一次性代码(只执行一次) dispatch_...

  • GCD简单使用

    队列类型 并发队列(Concurrent Dispatch Queue)任务并发(同步)执行 串行队列(Seria...

  • 如何取消GCD任务

    转载自:教你如何取消GCD任务 GCD 是一种非常方便的使用多线程的方式。通过使用 GCD,我们可以在确保尽量简单...

  • GCD的简单使用

    什么是GCD 全称是Grand Central Dispatch,中枢调度器纯C语言,提供了非常多强大的函数 GC...

  • GCD的简单使用

    GCD简单使用 GC使用就2个步骤 定制任务 将任务添加到队列中,gcd会自动将队列中的任务取出,放到对应的线程中...

网友评论

  • LeeDev:你用上面的 那段 打印一下代码
    2016-05-30 16:17:20.240 GCD的同步操作[9572:164665] 第1个
    2016-05-30 16:17:20.286 GCD的同步操作[9572:164604] 最后 一个
    2016-05-30 16:17:23.478 GCD的同步操作[9572:164667] 第3个
    2016-05-30 16:17:23.670 GCD的同步操作[9572:164666] 第4个
    2016-05-30 16:17:23.703 GCD的同步操作[9572:164669] 第5个
    2016-05-30 16:17:23.786 GCD的同步操作[9572:164668] 第2个
    LeeDev:@CoderQun 我刚试了下 打印的 结果 是
    2016-05-30 17:42:15.123 GCD的同步操作[10531:204820] 最后 一个
    2016-05-30 17:42:15.091 GCD的同步操作[10531:204923] 第1个
    2016-05-30 17:42:18.234 GCD的同步操作[10531:204925] 第2个
    2016-05-30 17:42:18.297 GCD的同步操作[10531:204927] 第5个
    2016-05-30 17:42:18.320 GCD的同步操作[10531:204922] 第4个
    2016-05-30 17:42:18.341 GCD的同步操作[10531:204924] 第3个
    马戏团小丑:@lichongyang_arc 之前开发中也遇到这样子的问题 很奇葩 但是也没找到原因 后来用下面那个enter leave 就不会出现问题
    LeeDev:@lichongyang_arc 顺序 不是 我们 理想的 应该 “最后一个” 应该显示在 最后面。
  • LeeDev:- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    dispatch_group_t group = dispatch_group_create();
    dispatch_group_async(group,dispatch_get_global_queue(0, 0), ^{

    [NSThread detachNewThreadSelector:@selector(firstThread) toTarget:self withObject:nil];

    // dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(4 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
    // NSLog(@"第1个");
    // });

    });
    dispatch_group_async(group,dispatch_get_global_queue(0, 0), ^{

    [NSThread detachNewThreadSelector:@selector(secondThread:) toTarget:self withObject:@"2"];

    });

    dispatch_group_async(group,dispatch_get_global_queue(0, 0), ^{

    [NSThread detachNewThreadSelector:@selector(secondThread:) toTarget:self withObject:@"3"];

    });

    dispatch_group_async(group,dispatch_get_global_queue(0, 0), ^{

    [NSThread detachNewThreadSelector:@selector(secondThread:) toTarget:self withObject:@"4"];

    });

    dispatch_group_async(group,dispatch_get_global_queue(0, 0), ^{

    [NSThread detachNewThreadSelector:@selector(secondThread:) toTarget:self withObject:@"5"];

    });

    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
    NSLog(@"最后 一个 ");
    });


    }


    - (void)firstThread {


    NSLog(@"第1个");

    }

    - (void)secondThread:(NSNumber *)num {

    NSInteger i = 1000000000;
    while (i--) {

    }

    NSString * message = [NSString stringWithFormat:@"第%@个",num];
    NSLog(@"%@",message);
    }

    - (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
    }

本文标题:GCD的简单使用

本文链接:https://www.haomeiwen.com/subject/emuqdttx.html