面试锦囊之多线程

😝😝😝

多线程相关的话题是面试过程中必不可少的话题。有些面试官可能是要你自己谈谈对多线程的认识,而有些则是出道题给你,让你直接手写或者让你口述实现。

谈谈对多线程的认识

在OC中实现多线程的方法有3种(NSThreadGCDNSOperation),
NSThread是苹果官方提供的,可以直接操作线程对象。不过需要程序员自己管理线程的生命周期(主要是创建),所以偶尔用用。比如 [NSThread currentThread],它可以获取当前线程信息,用于调试十分方便。而GCDNSOperation都是系统自动管理线程周期。

GCD是基于C底层的API,是一种更轻量级的,会自动合理地利用更多的CPU内核(比如双核、四核)。以FIFO(先进先出,后进后出)的顺序执行任务。GCD中的核心概念:任务 队列。任务:即操作,你想要干什么,说白了就是一段代码,在GCD中就是一个 Block,所以添加任务十分方便。任务有两种执行方式: 同步执行异步执行,他们之间的区别是是否会创建新的线程。有串行、并发、主队列、全局队列、组队列。其中队列任务组合的方式总共有7种,其中使用频率最高的是:并发队列+异步执行(多个任务同时执行并发执行,会开启多条线程)。关于GCD的一些其他具体组合使用方式,可查阅笔者之前的文章。点我查看

NSOperation是基于GCD更高一层的封装,相对于GCD更加强大。可以给operation之间添加依赖关系、取消一个正在执行的operation、暂停和恢复operationQueue等。关于NSOperation的其他一些知识点,可查阅笔者之前的文章。点我查看

多线程的面试题目

  • (1)A,B,C三个线程,要求执行完A,B后才能执行C,怎么做?

实现思路一: 添加依赖关系,A、B都依赖于C

     //1.创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];
    //2.创建操作
    NSBlockOperation *operationA = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"A----%@",[NSThread currentThread]); 
    }];
    
    NSBlockOperation *operationB = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"B----%@",[NSThread currentThread]);
    }];
    
    NSBlockOperation *operationC = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"C----%@",[NSThread currentThread]);        
    }];
    
    //3.添加依赖
    [operationC addDependency:operationA]; // 让C 依赖于 A,则先执行A,再执行C
    [operationC addDependency:operationB]; // 让C 依赖于 B,则先执行B,再执行C
    //4.添加操作到队列中
    [queue addOperation:operationA];
    [queue addOperation:operationB];
    [queue addOperation:operationC];
思路一打印结果.png

实现思路二:dispatch_group_notify

    dispatch_queue_t queue = dispatch_queue_create("label", DISPATCH_QUEUE_CONCURRENT);
    dispatch_group_t group = dispatch_group_create();
    dispatch_group_async(group, queue, ^{
         NSLog(@"A----%@",[NSThread currentThread]);
    });
    dispatch_group_async(group, queue, ^{
        NSLog(@"B----%@",[NSThread currentThread]);
    });
    //等前面的任务执行完毕后 会自动执行这个任务
    dispatch_group_notify(group, queue, ^{ 
         NSLog(@"C----%@",[NSThread currentThread]);
    });
思路二打印结果.png

实现思路三:dispatch_barrier_(a)sync

    dispatch_queue_t queue = dispatch_queue_create("label", DISPATCH_QUEUE_CONCURRENT);
    dispatch_async(queue, ^{
        NSLog(@"A----%@",[NSThread currentThread]);
    });
    dispatch_async(queue, ^{
        NSLog(@"B----%@",[NSThread currentThread]);
    });
 // dispatch_barrier_async: 在它前面的任务执行结束后它才执行,在它后面的任务等它执行完成后才会执行
    dispatch_barrier_async(queue, ^{
        NSLog(@"C----%@",[NSThread currentThread]);
    });
思路一打印结果.png
  • (2)结合使用AFNetworking多次请求,实现线程同步以及依赖。

在使用AFNetworking之前,我们首先看一段代码
需求:吃饭 睡觉 打豆豆 -> 任务完成

    dispatch_group_t group = dispatch_group_create();
    dispatch_queue_t queue = dispatch_queue_create("test", DISPATCH_QUEUE_CONCURRENT);
    
    dispatch_group_async(group, queue, ^{        
        NSLog(@"吃饭");
    });
    dispatch_group_async(group, queue, ^{
        NSLog(@"睡觉");
    });
    dispatch_group_async(group, queue, ^{
        NSLog(@"打豆豆");
    });
    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
        NSLog(@"任务完成");
    });
图片.png

那么使用AFNetworking之后呢,会触发怎么的结果呢?我们一起拭目以待。

- (void)requestFormData{
    
    
    dispatch_group_t group = dispatch_group_create();
    dispatch_queue_t queue = dispatch_queue_create("test", DISPATCH_QUEUE_CONCURRENT);
    
    dispatch_group_async(group, queue, ^{
        [self requestWithType:@"吃饭"];
    });

    dispatch_group_async(group, queue, ^{
        [self requestWithType:@"睡觉"];
    });
    
    dispatch_group_async(group, queue, ^{
        [self requestWithType:@"打豆豆"];
    });

    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
        NSLog(@"任务完成");
    });
}
- (void)requestWithType:(NSString *)type{

    AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
    [manager POST:@"http://www.mocky.io/v2/5b6685533200006a00ee11b1" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
              
        NSLog(@"%@",type);
        
    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {}];    
}
图片.png

综上打印,你心里可能会产生一个疑惑,只是换成了AFNetworking,打印结果却发生了变化,用过AFNetworking的伙伴们应该都知道在进行网络请求的时候内部是又开了线程的,那么这时候我们应该怎么实现吃饭睡觉打豆豆这个需求呢,请接着继续往下看。

方式一:使用信号量进行破解

- (void)requestWithType:(NSString *)type{
    
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
    AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
    [manager POST:@"http://www.mocky.io/v2/5b6685533200006a00ee11b1" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {       

         long flag  =   dispatch_semaphore_signal(semaphore); 
        NSLog(@"%@ --%ld",type,flag);

    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {

        dispatch_semaphore_signal(semaphore);
    }];
    
    long flag  =  dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
  
    NSLog(@"%@ --%ld",type,flag);
}

图片.png

方式二:dispatch_group_enterdispatch_group_leave配合使用

- (void)requestFormData{
    
    
    dispatch_group_t group = dispatch_group_create();
    dispatch_queue_t queue = dispatch_queue_create("test", DISPATCH_QUEUE_CONCURRENT);
    
    dispatch_group_async(group, queue, ^{
        [self requestWithType:@"吃饭" group:group];        
    });

    dispatch_group_async(group, queue, ^{
        [self requestWithType:@"睡觉" group:group];
    });
    
    dispatch_group_async(group, queue, ^{
        [self requestWithType:@"打豆豆" group:group];
    });
    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
       
        NSLog(@"任务完成");
    });
}
- (void)requestWithType:(NSString *)type group:(dispatch_group_t)group{
    
//通知group,下面的任务马上要放到group中执行了
     dispatch_group_enter(group);
    AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
    [manager POST:@"http://www.mocky.io/v2/5b6685533200006a00ee11b1" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
       
//通知group,任务完成了,该任务要从group中移除了
        dispatch_group_leave(group);
        NSLog(@"%@",type);
        
    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
       dispatch_group_leave(group);
    }];
}
图片.png

如果上面对你来说太so seay了,于是在此基础添加一个要求,吃完饭后,打会豆豆,之后便允许睡觉,才算完成任务了。革命尚未成功,同志仍需努力😁😁
使用AFNetworking请求,实现线程同步以及依赖

- (void)requestFormData{
    
    NSBlockOperation *operation1 = [NSBlockOperation blockOperationWithBlock:^{
        [self requestWithType:@"吃饭"];
    }];
    
    NSBlockOperation *operation2 = [NSBlockOperation blockOperationWithBlock:^{
        [self requestWithType:@"睡觉"];
    }];
    
    NSBlockOperation *operation3 = [NSBlockOperation blockOperationWithBlock:^{
        [self requestWithType:@"打豆豆"];
    }];
    
    NSBlockOperation *operation4 = [NSBlockOperation blockOperationWithBlock:^{
        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            NSLog(@"任务完成");
        }];
    }];

    [operation2 addDependency:operation1];      
    [operation4 addDependency:operation1];      
    [operation4 addDependency:operation2];      
    [operation4 addDependency:operation3];
 
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    [queue addOperations:@[operation1,operation2, operation3, operation4] waitUntilFinished:NO];

}
- (void)requestWithType:(NSString *)type {
  dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
    AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
    [manager POST:@"http://www.mocky.io/v2/5b6685533200006a00ee11b1" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
       
        NSLog(@"%@",type);
        
    dispatch_semaphore_signal(semaphore);
        
        
    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
        dispatch_semaphore_signal(semaphore);

    }];
    
    dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
}

图片.png

知识点补给站

  • addDependency不能添加相互依赖,例如:A依赖B,B依赖A,这样会导致死锁

  • 创建信号量,可以设置信号量的资源数。0表示没有资源。如:dispatch_semaphore_t semaphore = dispatch_semaphore_create(0)

  • 等待信号,会让信号量值减一,当信号量值为0时会等待(直到超时),否则正常执行;如:dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER)

  • 发送一个信号,会让信号总量加1。如:dispatch_semaphore_signal(semaphore)


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

推荐阅读更多精彩内容

  • 本文用来介绍 iOS 多线程中 GCD 的相关知识以及使用方法。这大概是史上最详细、清晰的关于 GCD 的详细讲...
    花花世界的孤独行者阅读 499评论 0 1
  • 转载自:http://www.jianshu.com/p/665261814e24 谈到iOS多线程,一般都会谈到...
    CodingMann丶许半仙阅读 481评论 0 0
  • 一、多线程简介: 所谓多线程是指一个 进程 -- process(可以理解为系统中正在运行的一个应用程序)中可以开...
    寻形觅影阅读 1,026评论 0 6
  • ​ 前言:这可能是史上最全面的一篇iOS 多线程博客了(王婆卖瓜一番😄),从多线程的基本概念,进程的概念,引出i...
    阿饼six阅读 848评论 2 11
  • 正值春运,人的神经会趋于同步的感觉,出门即掂记会较平日紧张,车流趋密、人流趋织。于是乎,人没有在机场坐定,就会往前...
    c29b18170e8a阅读 212评论 0 0