GCD的简单使用

GCD简单使用

  • GC使用就2个步骤
  1. 定制任务
  2. 将任务添加到队列中,gcd会自动将队列中的任务取出,放到对应的线程中执行,任务遵循队列:先进先出,后进后出 ,2个口 。。。。栈内存:先进后出,后进先出,一个口

gcd2个用来执行任务的常用函数

//异步 queue 队列 block 任务
dispatch_async(dispatch_queue_t queue, dispatch_block_t block);
//同步
dispatch_sync(dispatch_queue_t queue, dispatch_block_t block);
//这一句代码做了2件事 ,1,封装任务 ,2,把任务丢到队列中

gcd 队列分2大类。

队列的作用 1:队列就是用来装任务的,并且安排队列的任务到那个线程中执行 , 封装任务 , 2:让任务到线程中执行

  1. 并发队列 2.串行队列

并发队列
1. 可以让多个任务并发(同时)执行(自动开启多个线程同时执行任务)
2. 并发功能只能在异步函数先才有效(dispatch_async)

串行队列
1.任务一个接一个的执行

同步异步 的区别

  • 同步:只能在当前线程中执行任务,不具备开新线程的能力
  • 异步 :可以在新的线程中执行任务,具备开启新线程的能力

术语之间的区别于影响,同步,异步,并发,串行

同步函数(dispatch_sync)异步函数(dispatch_async)主要影响能否开启新的线程 , 同步函数,立刻马上执行。他不执行完就不能进行下一条

同步函数:只能在当前线程中执行任务,执行完成后进行下一条任务,不具备开新线程的能力
异步函数 :可以在新的线程中执行任务,具备开启新线程的能力

串行队列和并发队列主要影响任务的执行方式

并发队列:允许多个任务并发执行
串行串行:一个任务完成才能执行下一个任务

一些组合例子

创建 异步函数 + 并发队列

// 创建 异步函数 + 并发队列:次方法可以开多条线程,队列任务是异步(即并发)执行的
-(void)asyncConcurrent
{
    /*
     一 ,创建队列
     一个队列可以添加多个任务
     1,参数一:C语言字符串,随便给他命名
     2,参数二:队列的类型  DISPATCH_QUEUE_CONCURRENT 并发队列
                        DISPATCH_QUEUE_SERIAL 串行队列
     */
    dispatch_queue_t queue = dispatch_queue_create("xc.download", DISPATCH_QUEUE_CONCURRENT);
    
    //二,封装任务
    /*
        参数1: 队列
        参数2: 要执行的任务
     */
    dispatch_async(queue, ^{
        NSLog(@"NSThread1 = , %@" ,[NSThread currentThread]);
    });
    
    dispatch_async(queue, ^{
        NSLog(@"NSThread2 = , %@" ,[NSThread currentThread]);
    });
}

异步函数 + 串行队列

// 异步函数 + 串行队列 ;是可以开线程的,只开一条线程,是串行执行的
-(void)asyncSerial
{
   //创建串行队列
   dispatch_queue_t queue = dispatch_queue_create("xc.download", DISPATCH_QUEUE_SERIAL);
  
   //异步函数
   dispatch_async(queue, ^{
       NSLog(@"NSThread1 = , %@" ,[NSThread currentThread]);
   });
   dispatch_async(queue, ^{
       NSLog(@"NSThread2 = , %@" ,[NSThread currentThread]);
   });
}

同步函数 + 并发队列

//同步函数 + 并发队列 ;不会开线程的,因为是同步函数,
//同步函数不具备开线程的能力,所以任务是串行执行的
-(void)syncConcurrent
{
    //创建并发队列
    dispatch_queue_t queue = dispatch_queue_create("xc.download", DISPATCH_QUEUE_CONCURRENT);
    
    //同步函数
    dispatch_sync(queue, ^{
        NSLog(@"NSThread1 = , %@" ,[NSThread currentThread]);
    });
    dispatch_sync(queue, ^{
        NSLog(@"NSThread2 = , %@" ,[NSThread currentThread]);
    });
}

同步函数 + 串行队列

//同步函数 + 串行队列 ;
//不会开线程的,因为是同步函数,同步函数不具备开线程的能力,所以任务是串行执行的
-(void)syncSerial
{
  //创建串行队列
  dispatch_queue_t queue = dispatch_queue_create("xc.download", DISPATCH_QUEUE_SERIAL);
  
  //异步函数
  dispatch_sync(queue, ^{
      NSLog(@"NSThread1 = , %@" ,[NSThread currentThread]);
  });
  dispatch_sync(queue, ^{
      NSLog(@"NSThread2 = , %@" ,[NSThread currentThread]);
  });
}

只有:异步函数 + 并发队列:次方法可以开多条线程,队列任务是异步(即并发)执行的,这样才是并发执行。其余几种组合都是串行执行
gcd里面开多少条线程不是我们决定的,由系统决定

全局并发队列

 //获得全局队列
    /*
        1,第一个参数:优先级
  系统原来就带有的 ,
       2,第二个参数暂时没什么用,传0 即可
     */
    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

GCD主队列

  1. 主队列是GCD自带的一种特殊的串行队列
  2.只要在主队列任务,都会放到主线程中去执行
  3.使用     dispatch_get_main_queue()获得主队列

队列特点是

  1. 封装任务, 2. 安排任务在哪个线程中去执行。

主队列特点

主队列,会把他放到主线程中去执行,如果主队列有任务在执行,
那么主队列会暂停调用队列中的任务,知道主线程空闲为止

异步函数 + 主队列

//异步函数 + 主队列 。 不会开线程,所有任务主线程中去执行
-(void)asyncMain
{
  //获取主队列
  dispatch_queue_t queue = dispatch_get_main_queue() ;
  //异步函数
  dispatch_async(queue, ^{
      NSLog(@"NSThread1 = , %@" ,[NSThread currentThread]);
  });
  dispatch_async(queue, ^{
      NSLog(@"NSThread2 = , %@" ,[NSThread currentThread]);
  });
}
注意:同步函数 + 主队列 ,如果 在主线程中调用会报 错误 。 只能开启一条子线程,在 调用 同步函数 + 主队列 就可以了
//同步函数 + 主队列 。 错误。会造成死锁,。 因为主队列只能在主线程中执行,,

//如果 主线程 调用syncMain,方法会死锁,
因为调用方法发现 dispatch_sync 是同步函数,同步函数的任务放到  
主队列,然后因为是主队列,会把他放到主线程中去执行,而如果主线程调  
用了syncMain方法,里面是同步函数,同步函数在等主线程任务执行  
结束,而主线程在等同步函数执行完毕,这样就造成了死锁。所以要用  
子线程去调用
*/

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
//开启子线程调用
    [NSThread detachNewThreadSelector:@selector(syncMain) toTarget:self withObject:nil];
}
-(void)syncMain
{
    //获取主队列
    dispatch_queue_t queue = dispatch_get_main_queue() ;
    //同步函数
    dispatch_sync(queue, ^{
        NSLog(@"NSThread1 = , %@" ,[NSThread currentThread]);
    });
    dispatch_sync(queue, ^{
        NSLog(@"NSThread2 = , %@" ,[NSThread currentThread]);
    });
}

总结下:只有在并发队列和异步函数,才会开启新线程。
1,并发队列+异步函数 = 有开启新线程,并发执行任务
2,串行队列+异步函数 = 有开一天新线程,串行执行
3,其余情况都是,不会开启新线程,串行执行 (例:并发队列,同步函数 or串行队列 + 异步函数 等)

GCD 线程间的通信

   //GCD间的线程通信
    //创建子线程下载图片
    //获得全局主队列
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        //子线程下载图片
        NSString *str =  @"http://static.firefoxchina.cn/img/201710/4_59e999c8e6aa50.jpg";
        NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:str]];
        UIImage *image = [UIImage imageWithData:data];
        NSLog(@"%@" ,[NSThread currentThread]);
       
        //回到主线程更新UI ,注意 同步函数 和 主队列。死锁情况哟
        dispatch_async(dispatch_get_main_queue(), ^{
            self.imageview.image = image ;
            NSLog(@"%@" ,[NSThread currentThread]);
        });
    });
    

GCD一些常用方法

  1. 延迟调用方法
    /**
     方法一:延时调用改方法
     参数1: 调用的方法
     参数2:方法的参数
    参数3:延时的时间
     */
    [self performSelector:@selector(run:) withObject:nil afterDelay:2.0];


/**
     方法二
     延时2s调用 run方法
     @param run: 方法
     */
    [NSTimer scheduledTimerWithTimeInterval:2.0f target:self selector:@selector(run:) userInfo:nil repeats:YES];

  /**
    方法三
        延时2s
     DISPATCH_TIME_FOREVER
     @param DISPATCH_TIME_NOW ->从现在开始
     @param int64_t 延迟的时间
     @return
dispatch_get_main_queue 我们可以改变队列,来改变延时后在那个线程中执行
     */
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0f * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        
    });

栅栏函数

并发队列多个任务,那么我们想要某个任务,最后执行怎么办 ,所以用栅栏函数
注意:栅栏函数不能使用 全局并发队列,不然栅栏函数没有效果

//栅栏函数
-(void)barriermethod
{
    //创建并发队列 。栅栏函数注意不能用全局并发队列
    dispatch_queue_t queue = dispatch_queue_create("xc_queue", DISPATCH_QUEUE_CONCURRENT);
    //异步函数 。 3个人任务并发执行,无顺序 。 那么怎么给他们安排顺序呢。 ----->栅栏函数。
    dispatch_async(queue, ^{
        for (int i = 0; i< 100; i++) {
            NSLog(@"downLoad1 = %@" ,[NSThread currentThread]);
        }
    });
    dispatch_async(queue, ^{
        for (int i = 0; i< 100; i++) {
            NSLog(@"downLoad2 = %@" ,[NSThread currentThread]);
        }
    });
    //我们需要任务3最后执行。所以我们用栅栏函数
    //栅栏
    dispatch_barrier_async(queue, ^{
        NSLog(@"x-x-xx");
    });
    dispatch_async(queue, ^{
        NSLog(@"downLoad3 = %@" ,[NSThread currentThread]);
    });
}

GCD的快速迭代 ,迭代就是遍历的意思

for 循环就是迭代。 for循环是同步的

//dispatch快速迭代
-(void)apply
{
    /**
     快速迭代, dispatch_apply 会开启子线程去遍历。
     @param iterations#> 迭代的次数 description#>
     @param queue#> 队列。只能传并发队列,传串行队列就和for一样了没有意义,不能传主队列,可以全局并发队列 description#>
     @param size_t index 索引
     */
    dispatch_apply(10, dispatch_get_global_queue(0, 0), ^(size_t index) {
        NSLog(@"%zd --%@",index,[NSThread currentThread]);
    });
}
/* 打印结果。有多个线程
2017-10-23 23:33:19.605 NSTread基本使用[1293:46453] 2 --<NSThread: 0x6000002603c0>{number = 4, name = (null)}
2017-10-23 23:33:19.605 NSTread基本使用[1293:46418] 0 --<NSThread: 0x608000079c40>{number = 1, name = main}
2017-10-23 23:33:19.606 NSTread基本使用[1293:46455] 3 --<NSThread: 0x6080002676c0>{number = 5, name = (null)}
2017-10-23 23:33:19.606 NSTread基本使用[1293:46453] 4 --<NSThread: 0x6000002603c0>{number = 4, name = (null)}
2017-10-23 23:33:19.606 NSTread基本使用[1293:46418] 5 --<NSThread: 0x608000079c40>{number = 1, name = main}
2017-10-23 23:33:19.606 NSTread基本使用[1293:46455] 6 --<NSThread: 0x6080002676c0>{number = 5, name = (null)}
2017-10-23 23:33:19.606 NSTread基本使用[1293:46453] 7 --<NSThread: 0x6000002603c0>{number = 4, name = (null)}
2017-10-23 23:33:19.606 NSTread基本使用[1293:46418] 8 --<NSThread: 0x608000079c40>{number = 1, name = main}
2017-10-23 23:33:19.606 NSTread基本使用[1293:46455] 9 --<NSThread: 0x6080002676c0>{number = 5, name = (null)}
2017-10-23 23:33:19.605 NSTread基本使用[1293:46452] 1 --<NSThread: 0x600000260380>{number = 3, name = (null)}
*/

队列组。

队列组里面的任务的进行情况。我们可以知道。是否完成,如果完成可以拿到这个完成事件

//队列组 ,简写步骤 1
-(void)groupMethod
{
    //1.创建队列
    dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
    //2.创建队列组
    dispatch_group_t group = dispatch_group_create() ;
    
    //3.异步函数创建,并队列组管理
    dispatch_group_async(group, queue, ^{
        for (int i = 0; i < 100; i++) {
            NSLog(@"downLoad1 = %@ , i = %d" ,[NSThread currentThread],i);
        }
    });
    dispatch_group_async(group, queue, ^{
        for (int i = 0; i < 100; i++) {
            NSLog(@"downLoad2 = %@ , i = %d" ,[NSThread currentThread],i);
        }
    });
    //离开通知,监听到
    dispatch_group_notify(group, queue, ^{
        NSLog(@"队列组任务完成。离开,在这里说明上面任务都完成了");
    });
    //和上面dispatch_group_notify 效果一样,特点队列组的任务完成后他才会执行
    //dispatch_group_wait(group, DISPATCH_TIME_FOREVER);

}

比较老一点的队列组写法

//队列组 第二种
-(void)groupMethod2
{
    //1.创建队列
    dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
    //2.创建队列组
    dispatch_group_t group = dispatch_group_create() ;
    
    //3.在该方法后面的异步任务会被纳入到队列的监听范围
    dispatch_group_enter(group);
    //4.异步函数创建,并队列组管理
    dispatch_async(queue, ^{
        for (int i = 0; i< 100; i++) {
            NSLog(@"downLoad1 = %@" ,[NSThread currentThread]);
        }
        //5. 告诉队列组任务执行完毕了,离开群组  dispatch_group_leave(group);
 
        dispatch_group_leave(group);
    });
    
    //dispatch_group_enter(group);   dispatch_group_leave(group);配对的,必须以前出现
    dispatch_group_enter(group);
    dispatch_async(queue, ^{
        for (int i = 0; i< 100; i++) {
            NSLog(@"downLoad2 = %@" ,[NSThread currentThread]);
        }
        dispatch_group_leave(group);
    });
    //离开通知,监听到
    dispatch_group_notify(group, queue, ^{
        NSLog(@"队列组任务完成。离开,这个 内部是异步的");
    });
}
学习记录:如有不妥。请大神指出,😁😄。

NSOperation简单的入门
NSThread概念以及入门

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

推荐阅读更多精彩内容