iOS-多线程-(小码哥底层原理笔记)

2023-12-26 07:38

本文主要是介绍iOS-多线程-(小码哥底层原理笔记),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

iOS中常见的多线程方案

image.png

GCD的常用函数

同步方式执行任务

 dispatch_sync(dispatch_queue_t  _Nonnull queue, ^(void)block)

queue - 队列
block - 任务

  • 异步执行任务
dispatch_async(dispatch_queue_t  _Nonnull queue,^(void)block)

GCD的队列

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

  • 串行队列
    1.让任务一个接一个执行(一个任务完成后才能执行下个任务)

注意:在ARC环境下,GCD创建的队列不需要手动release,但是使用CF开头的带有create需要手动release。

  • (dispatch_sync和dispatch_async)同步函数和异步函数只能决定开启新线程的能力((dispatch_get_main_queue())主队列的任务一定是在主线程执行的)
  • 串行和并发主要影响的是任务串行执行还是并发执行

image.png
总结来说:

  • 同步函数(dispatch_sync)和主队列都不会开启新线程,并且任务都是串行执行
  • 串行队列(手动创建的串行队列和主队列)都是串行执行任务的
  • 只有在异步函数中使用并发队列才会开启新线程同时并发执行任务

产生死锁

1.下面代码会不会产生死锁?

//代码1
- (void)viewDidLoad {[super viewDidLoad];NSLog(@"任务1");dispatch_queue_t queue = dispatch_get_main_queue();dispatch_sync(queue, ^{NSLog(@"任务2");});NSLog(@"任务3");
}

答案:会产生死锁。viewDidLoad在主线程中执行,主队列中有任务viewDidLoad和任务2,根据队列先进先出的特点,任务2 必须要等到viewDidLoad执行完才能执行,同时因为执行任务2是在dispatch_sync中,根据dispatch_sync立马在当前线程执行任务,执行完毕后才能执行后续任务的特点,所以任务3在等任务2执行完,任务2在等viewDidLoad执行完,viewDidLoad要执行完必须是任务1,2,3都执行完。
image.png

  • dispatch_sync:立马在当前线程执行任务,执行完毕后才能执行后续任务
  • 队列的特点:FIFO(先进先出)

2.以上代码将dispatch_sync改为dispatch_async将不会产生死锁:

//代码2
- (void)viewDidLoad {[super viewDidLoad];NSLog(@"任务%@",[NSThread currentThread]);dispatch_queue_t queue = dispatch_get_main_queue();dispatch_async(queue, ^{NSLog(@"任务2%@",[NSThread currentThread]);});NSLog(@"任务3%@",[NSThread currentThread]);
}

打印结果

2020-11-28 13:10:02.165360+0800 多线程[52082:602232] 任务1<NSThread: 0x6000014f0100>{number = 1, name = main}
2020-11-28 13:10:02.165654+0800 多线程[52082:602232] 任务3<NSThread: 0x6000014f0100>{number = 1, name = main}
2020-11-28 13:10:02.180487+0800 多线程[52082:602232] 任务2<NSThread: 0x6000014f0100>{number = 1, name = main}

3.以下代码会不会产生死锁?

- (void)viewDidLoad {[super viewDidLoad];dispatch_queue_t queue = dispatch_queue_create("Queue", DISPATCH_QUEUE_SERIAL);NSLog(@"任务1");dispatch_async(queue, ^{//block1NSLog(@"任务2");dispatch_sync(queue, ^{//block2NSLog(@"任务3");});NSLog(@"任务4");});NSLog(@"任务5");
}

答:会产生死锁。block1,block2都加入到了串行队列中,同时block2使用的是dispatch_sync,所以需要立马执行block2,但是block2必须要等到block1执行完后才能拿出来执行,所以产生了死锁。
4.将以上代码转换为将不会产生死锁

- (void)viewDidLoad {[super viewDidLoad];dispatch_queue_t queue = dispatch_queue_create("Queue", DISPATCH_QUEUE_SERIAL);dispatch_queue_t queue2 = dispatch_queue_create(@"Queue1", DISPATCH_QUEUE_SERIAL);NSLog(@"任务1");dispatch_async(queue, ^{NSLog(@"任务2");dispatch_sync(queue2, ^{NSLog(@"任务3");});NSLog(@"任务4");});NSLog(@"任务5");
}

打印结果

2020-11-28 13:23:13.103083+0800 多线程[52260:611080] 任务1
2020-11-28 13:23:13.103243+0800 多线程[52260:611080] 任务5
2020-11-28 13:23:13.103257+0800 多线程[52260:611214] 任务2
2020-11-28 13:23:13.103347+0800 多线程[52260:611214] 任务3
2020-11-28 13:23:13.103415+0800 多线程[52260:611214] 任务4
  • 总结:使用dispatch_sync函数往当前串行队列中加入任务会产生死锁

多线程和runloop

5.以下代码打印的结果是什么?

 1. (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{dispatch_queue_t queue = dispatch_get_global_queue(0, 0);dispatch_async(queue, ^{NSLog(@"任务1");[self performSelector:@selector(test) withObject:nil afterDelay:.0];//用到定时器NSLog(@"任务3");});
}2. (void)test {NSLog(@"任务2");
}

打印结果:

2020-11-30 12:28:47.436127+0800 多线程[8844:154622] 1
2020-11-30 12:28:47.436291+0800 多线程[8844:154622] 3

答:没有打印任务2的原因是因为[self performSelector:@selector(test) withObject:nil afterDelay:.0]这句代码是往runloop中添加了定时器,同时通过dispatch_async在子线程中执行任务,而子线程中是默认没有开启runloop的,所以不执行任务2。如果像下面的代码一样启动了runloop,则会打印任务2。

 3. (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{dispatch_queue_t queue = dispatch_get_global_queue(0, 0);dispatch_async(queue, ^{NSLog(@"1");//这句代码是往runloop中添加了定时器[self performSelector:@selector(test) withObject:nil afterDelay:.0];NSLog(@"3");// [[NSRunLoop currentRunLoop]addPort:[[NSPort alloc]init] forMode:NSDefaultRunLoopMode];[[NSRunLoop currentRunLoop]runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];//[NSDate distantPast]: 可以表示的最早的时间//[NSDate distantFuture]:可以表示的最远的未来时间});
}4. (void)test {NSLog(@"2");
}

6.以下代码会打印什么?

 5. (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{NSThread *thread = [[NSThread alloc]initWithBlock:^{NSLog(@"任务1");}];[thread start];[self performSelector:@selector(test) onThread:thread withObject:nil waitUntilDone:YES];}6. (void)test {NSLog(@"任务2");
}

打印结果

2020-11-30 12:53:10.923568+0800 多线程[9289:175207] 任务1
2020-11-30 12:53:10.957516+0800 多线程[9289:174984] *** Terminating app due to uncaught exception 'NSDestinationInvalidException', reason: '*** -[ViewController performSelector:onThread:withObject:waitUntilDone:modes:]: target thread exited while waiting for the perform'
*** First throw call stack:

答:会打印任务1,然后崩溃。因为没有启动runloop,可改为以下代码:

 7. (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{NSThread *thread = [[NSThread alloc]initWithBlock:^{NSLog(@"任务1");
//        [[NSRunLoop currentRunLoop]addPort:[[NSPort alloc]init] forMode:NSDefaultRunLoopMode];[[NSRunLoop currentRunLoop]runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];}];[thread start];[self performSelector:@selector(test) onThread:thread withObject:nil waitUntilDone:YES];}8. (void)test {NSLog(@"任务2");
}

队列组的使用

7.如何用gcd实现以下功能
异步并发执行任务1,任务2
等任务1,任务2都执行完毕后,再回到主线程执行任务3

//创建队列组dispatch_group_t group = dispatch_group_create();//创建并发队列dispatch_queue_t queue = dispatch_queue_create("myqueue", DISPATCH_QUEUE_CONCURRENT);//添加异步任务dispatch_group_async(group, queue, ^{for (int i = 0 ;i < 5; i++) {NSLog(@"任务1-%@",[NSThread currentThread]);}});dispatch_group_async(group, queue, ^{for (int i = 0 ;i < 5; i++) {NSLog(@"任务2-%@",[NSThread currentThread]);}});//等前面的任务执行完成后,会自动执行这个任务// dispatch_group_notify(group, queue, ^{//回到主线程//  dispatch_async(dispatch_get_main_queue(), ^{//     for (int i = 0 ;i < 5; i++) {//       NSLog(@"任务3-%@",[NSThread currentThread]);// }//});//});dispatch_group_notify(group, dispatch_get_main_queue(), ^{//回到主线程for (int i = 0 ;i < 5; i++) {NSLog(@"任务3-%@",[NSThread currentThread]);}});

8.上述代码改为不回主线程,继续异步执行任务:

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{//创建队列组dispatch_group_t group = dispatch_group_create();//创建并发队列dispatch_queue_t queue = dispatch_queue_create("myqueue", DISPATCH_QUEUE_CONCURRENT);//添加异步任务dispatch_group_async(group, queue, ^{for (int i = 0 ;i < 5; i++) {NSLog(@"任务1-%@",[NSThread currentThread]);}});dispatch_group_async(group, queue, ^{for (int i = 0 ;i < 5; i++) {NSLog(@"任务2-%@",[NSThread currentThread]);}});//等前面的任务执行完成后,会自动执行这个任务dispatch_group_notify(group, queue, ^{//回到主线程for (int i = 0 ;i < 5; i++) {NSLog(@"任务3-%@",[NSThread currentThread]);}});dispatch_group_notify(group, queue, ^{//回到主线程for (int i = 0 ;i < 5; i++) {NSLog(@"任务4-%@",[NSThread currentThread]);}});
}

等任务1和任务2都执行完毕后,再交替执行任务3和任务4

参考链接:https://ke.qq.com/course/package/11609?quicklink=1

这篇关于iOS-多线程-(小码哥底层原理笔记)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/538585

相关文章

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Redis主从复制的原理分析

《Redis主从复制的原理分析》Redis主从复制通过将数据镜像到多个从节点,实现高可用性和扩展性,主从复制包括初次全量同步和增量同步两个阶段,为优化复制性能,可以采用AOF持久化、调整复制超时时间、... 目录Redis主从复制的原理主从复制概述配置主从复制数据同步过程复制一致性与延迟故障转移机制监控与维

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R

Redis主从复制实现原理分析

《Redis主从复制实现原理分析》Redis主从复制通过Sync和CommandPropagate阶段实现数据同步,2.8版本后引入Psync指令,根据复制偏移量进行全量或部分同步,优化了数据传输效率... 目录Redis主DodMIK从复制实现原理实现原理Psync: 2.8版本后总结Redis主从复制实

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

hdu4407(容斥原理)

题意:给一串数字1,2,......n,两个操作:1、修改第k个数字,2、查询区间[l,r]中与n互质的数之和。 解题思路:咱一看,像线段树,但是如果用线段树做,那么每个区间一定要记录所有的素因子,这样会超内存。然后我就做不来了。后来看了题解,原来是用容斥原理来做的。还记得这道题目吗?求区间[1,r]中与p互质的数的个数,如果不会的话就先去做那题吧。现在这题是求区间[l,r]中与n互质的数的和

安卓链接正常显示,ios#符被转义%23导致链接访问404

原因分析: url中含有特殊字符 中文未编码 都有可能导致URL转换失败,所以需要对url编码处理  如下: guard let allowUrl = webUrl.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) else {return} 后面发现当url中有#号时,会被误伤转义为%23,导致链接无法访问

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

【编程底层思考】垃圾收集机制,GC算法,垃圾收集器类型概述

Java的垃圾收集(Garbage Collection,GC)机制是Java语言的一大特色,它负责自动管理内存的回收,释放不再使用的对象所占用的内存。以下是对Java垃圾收集机制的详细介绍: 一、垃圾收集机制概述: 对象存活判断:垃圾收集器定期检查堆内存中的对象,判断哪些对象是“垃圾”,即不再被任何引用链直接或间接引用的对象。内存回收:将判断为垃圾的对象占用的内存进行回收,以便重新使用。