学习总结-GDC

多线程(GCD)

一.实现方式

  • pthread: C语言实现,不常用
  • nsthred: 基本上也不用
  • gcd: 常用(个人常用)
  • NSOperation(对gcd的封装)

二.生命周期(五种状态)

状态图
  • 新建:实例化线程对象
  • 就绪:向线程对象发送start消息,线程对象被加入可调度线程池等待CPU调度。
  • 运行:CPU 负责调度可调度线程池中线程的执行。线程执行完成之前,状态可能会在就绪和运行之间来回切换。就绪和运行之间的状态变化由CPU负责,程序员不能干预。
  • 阻塞:当满足某个预定条件时,可以使用休眠或锁,阻塞线程执行。sleepForTimeInterval(休眠指定时长),sleepUntilDate(休眠到指定日期),@synchronized(self):(互斥锁)。
  • 死亡:正常死亡,线程执行完毕。非正常死亡,当满足某个条件后,在线程内部中止执行/在主线程中止线程对象
  • 还有线程的exitcancel
  • [NSThread exit]:一旦强行终止线程,后续的所有代码都不会被执行。
  • [thread cancel]取消:并不会直接取消线程,只是给线程对象添加 isCancelled 标记。

三.多线程安全问题

原因: 多个线程同时访问一个资源所引起的数据错乱问题

解决方案
  • 同步锁(互斥锁):
@synchronized(锁对象(一般用self)) {
    // 需要锁定的代码
}
  • 自旋锁(耗性能) :加了自旋锁,当新线程访问代码时,如果发现有其他线程正在锁定代码,新线程会用死循环的方式,一直等待锁定的代码执行完成 .atomic修饰的带有自旋锁,所以说是线程安全的.但是取值的时候是可以有多个线程同时取值的,nonatomic 是线程不安全的,但是效率高,一般使用 nonatomic

关于GCD

1: GCD特点

  • GCD会自动利用CPU的多核特性
  • GCD自动管理线程的生命周期,开发人员只需告诉GCD如何执行,执行什么任务

2: GCD的基本概念

  • 队列: 装载线程任务的队形结构(先进先出),有两种队列穿行队列并发队列

3: 队列的创建方法

dispatch_queue_t queue = dispatch_queue_create(const char * _Nullable label,  
dispatch_queue_attr_t  _Nullable attr)

参数说明:第一个为标识符,可不传;第二个参数为队列类型(并发队列DISPATCH_QUEUE_CONCURRENT,穿行队列DISPATCH_QUEUE_SERIAL);
此外还有两种创建队列的方法

dispatch_queue_t queue = dispatch_get_global_queue(0, 0);//获取全局并发队列
dispatch_queue_t queue = dispatch_get_main_queue();//获取主队列

4: 任务的创建方式

  • 同步:dispatch_sync
  • 异步:dispatch_async
    dispatch_sync(dispatch_get_global_queue(0, 0), ^{// 同步执行任务
        NSLog(@"我是同步执行的任务");
    });
    
    dispatch_async(dispatch_get_global_queue(0, 0), ^{// 异步执行任务
        NSLog(@"我是异步执行的任务");
    });

5: GCD的使用
组合方式:有六种组合方式(串行/并发/主队列 ✖️同步/异步)

  1. 串行同步
  2. 串行异步
  3. 并发同步
  4. 并发异步
  5. 主队列同步
  6. 主队列异步

同步/异步决定是否开新线程,串行/并发/主队列决定是否按顺序执行任务
特别说明:5.主队列同步,会发生死锁,程序奔溃

- (void)testSync {
    dispatch_sync(dispatch_get_main_queue(), ^{
        //需要执行的任务
    });
}

原因:在主线程中使用同步对于任务是立刻执行的,即:block中的任务会立刻执行,但是此时主线程正在处理testSync()方法,所以任务就需要等待testSync()执行后才能执行,但是该任务就处于testSync()方法中,所以就造成无限等待中也就是死锁

GCD实用API

  • dispatch_after:延时一定时间后处理任务
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)),dispatch_get_main_queue(), ^{
        //3秒后需要做的事情
    });
  • dispatch_barrier_async:栅栏函数,用于分割任务的执行顺序
dispatch_queue_t asyncQueue = dispatch_queue_create(nil, DISPATCH_QUEUE_CONCURRENT);
    dispatch_async(asyncQueue, ^{
        for (NSUInteger i = 0; i < 10; i++) {
            NSLog(@"11111--%zd--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_async(asyncQueue, ^{
        for (NSUInteger i = 0; i < 10; i++) {
            NSLog(@"2222--%zd--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_barrier_async(asyncQueue, ^{
        NSLog(@"我在这里把他拦截下来");
    });
    
    dispatch_async(asyncQueue, ^{
        for (NSUInteger i = 0; i < 10; i++) {
            NSLog(@"33333--%zd--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_async(asyncQueue, ^{
        for (NSUInteger i = 0; i < 10; i++) {
            NSLog(@"44444--%zd--%@",i,[NSThread currentThread]);
        }
    });

根据log可以发现,'3','4' 都会再'1','2'执行完之后才会执行。这样就达到了栅栏的效果。
注意: 栅栏函数不能使用全局并发队列, 必须是自己通过函数创建的并发队列,否则将达不到栅栏效果

  • dispatch_apply:快速迭代,直接看效果
CFTimeInterval startTime = CFAbsoluteTimeGetCurrent();
//    for (int i = 0; i < 10000; i ++) {
//        NSLog(@"");
//    }//2.395429
    dispatch_queue_t concurrentQ = dispatch_queue_create(nil, DISPATCH_QUEUE_CONCURRENT);
    dispatch_apply(10000, concurrentQ, ^(size_t i) {
        NSLog(@"");
    });//2.40
    CFTimeInterval endTime = CFAbsoluteTimeGetCurrent();
    NSLog(@"%f",endTime-startTime);

可能是我在循环中没有做什么操作吧,for循环竟然更快dispatch_applydispatch_sync函数和dispatch_group的关联API

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

推荐阅读更多精彩内容