多线程
你们项目中为什么多线程用GCD而不用NSOperation呢? 你有没有发现国外的大牛他们多线程都是用NSOperation? 你能告诉我他们这样做的理由吗?
关系:
①:先搞清两者的关系,NSOpertaionQueue用GCD构建封装的,是GCD的高级抽象!
②:GCD仅仅支持FIFO队列,而NSOperationQueue中的队列可以被重新设置优先级,从而实现不同操作的执行顺序调整。GCD不支持异步操作之间的依赖关系设置。如果某个操作的依赖另一个操作的数据(生产者-消费者模型是其中之一),使用NSOperationQueue能够按照正确的顺序执行操作。GCD则没有内建的依赖关系支持。
③:NSOperationQueue支持KVO,意味着我们可以观察任务的执行状态。
了解以上不同,我们可以从以下角度来回答
性能:①:GCD更接近底层,而NSOperationQueue则更高级抽象,所以GCD在追求性能的底层操作来说,是速度最快的。这取决于使用Instruments进行代码性能分析,如有必要的话
②:从异步操作之间的事务性,顺序行,依赖关系。GCD需要自己写更多的代码来实现,而NSOperationQueue已经内建了这些支持
③:如果异步操作的过程需要更多的被交互和UI呈现出来,NSOperationQueue会是一个更好的选择。底层代码中,任务之间不太互相依赖,而需要更高的并发能力,GCD则更有优势
最后的一句话:别忘了高德纳的教诲:“在大概97%的时间里,我们应该忘记微小的性能提升。过早优化是万恶之源。”只有Instruments显示有真正的性能提升时才有必要用低级的GCD。
详解GCD死锁
unix上进程怎么通信?
- UNIX主要支持三种通信方式:
- 基本通信:主要用来协调进程间的同步和互斥
- 锁文件通信:通信的双方通过查找特定目录下特定类型的文件(称锁文件)来完成进程间 对临界资源访问时的互斥;例如进程p1访问一个临界资源,首先查看是否有一个特定类型文件,若有,则等待一段时间再查找锁文件。
- 记录锁文件
- 管道通信:适应大批量的数据传递
- IPC :适应大批量的数据传递
列举几种进程的同步机制、进程的通信途径、死锁及死锁的处理方法。
进程的同步机制原子操作 信号量机制 自旋锁 管程,会合,分布式系统
进程之间通信的途径:共享存储系统消息传递系统管道:以文件系统为基础
进程死锁的原因:资源竞争及进程推进顺序非法
死锁的4个必要条件:互斥、请求保持、不可剥夺、环路
死锁的处理:鸵鸟策略、预防策略、避免策略、检测与解除死锁
线程与进程的区别和联系?
线程是进程的基本单位。
进程和线程都是由操作系统所产生的程序运行的基本单元,系统利用该基本单元实现系统对应用的并发性。
进程和线程的主要差别在于它们是不同的操作系统资源管理方式。
进程有独立的地址空间,一个进程崩溃后,在?;つJ较?不会对其它进程产生影响。
线程只是一个进程中的不同执行路径。
线程有自己的堆栈和局部变量,但线程之间没有单独的地址空间,一个线程死掉就等于整个进程死掉,所以多进程的程序要比多线程的程序健壮,但在进程切换时,耗费资源较大,效率要差一些。
但对于一些要求同时进行并且又要共享某些变量的并发操作,只能用线程,不能用进程。
iOS线程间怎么通信?
- performSelector:onThread:withObject:waitUntilDone:
- NSMachPort
(基本机制:A线程(父线程)创建NSMachPort对象,并加入A线程的runloop。当创建B线程(辅助线程)时,将创建的NSMachPort对象传递到主体入口点,B线程(辅助线程)就可以使用相同的端口对象将消息传回A线程(父线程)。)
iOS多线程的底层实现?
- 首先搞清楚什么是线程、什么是多线程
- Mach是第一个以多线程方式处理任务的系统,因此多线程的底层实现机制是基于Mach的线程
- 开发中很少用Mach级的线程,因为Mach级的线程没有提供多线程的基本特征,线程之间是独立的
- 开发中实现多线程的方案
- C语言的POSIX接口:#include <pthread.h>
- OC的NSThread
- C语言的GCD接口(性能最好,代码更精简)
- OC的NSOperation和NSOperationQueue(基于GCD)
谈谈多线程安全问题的几种解决方案?何为线程同步,如何实现的?分线程回调主线程方法是什么,有什么作用?
- 解决方案: 使用锁: 锁是线程编程同步工具的基础。锁可以让你很容易?;ご胫幸淮罂榍蛞员隳憧梢匀繁4氲恼沸?。
- 使用POSIX互斥锁;
- 使用NSLock类;
- 使用@synchronized指令等。
- 回到主线程的方法: dispatch_async(dispatch_get_main_queue(), ^{ });
- 作用: 主线程是显示UI界面,子线程多数是进行数据处理
使用atomic一定是线程安全的吗?
不是的。
atomic原子操作,系统会为setter方法加锁。 具体使用 @synchronized(self){//code }
nonatomic不会为setter方法加锁。
atomic:线程安全,需要消耗大量系统资源来为属性加锁
nonatomic:非线程安全,适合内存较小的移动设备
使用atomic并不能保证绝对的线程安全,对于要绝对保证线程安全的操作,还需要使用更高级的方式来处理,比如NSSpinLock、@syncronized等
谈谈你对多线程开发的理解(多线程的好处,多线程的作用)?ios中有几种实现多线程的方法?
- 好处:
- 使用线程可以把占据时间长的程序中的任务放到后台去处理
- 用户界面可以更加吸引人,这样比如用户点击了一个按钮去触发某些事件的处理,可以弹出一个进度条来显示处理的进度
- 程序的运行效率可能提高
- 在一些等待的任务实现上如用户输入、文件读写和网络收发数据等,线程就比较有用了。
- 缺点:
- 如果有大量的线程,会影响性能,因为操作系统需要在它们之间切换。
- 更多的线程需要更多的内存空间。
- 线程的中止需要考虑其对程序运行的影响。
- 通??槟P褪菔窃诙喔鱿叱碳涔蚕淼?,需要防止线程死锁情况的发生。
- 实现多线程的方法:
- NSObject的类方法 // -(void)performSelectornBackground/OnMainThread:(SEL)aSelector withObject:(id)arg
- NSThread
- NSOperation
- GCD
OC中异步使用的哪种事件模型,iOS中异步实现机制
- 异步非阻塞 I/O(AIO)
详细谈谈GCD
-
推出的时间 iOS4 目的是用来取代NSThread(ios2.0推出)的,是 C语言框架,它能够自动利用更多CPU的核数,并且会自动管理线程的生命周期。
- CGD的两个核心概念:任务, 队列
- 任务:记为在block中执行的代码。
- 队列:用来存放任务的。
- 注意事项: 队列 != 线程。队列中存放的任务最后都要由线程来执行!。队列的原则:先进先出,后进后出(FIFO/ First In First Out)
-
队列又分为四种种:1 串行队列 2 并发队列 3 主队列 4 全局队列
- 串行队列: 任务一个接一个的执行。
- 并发队列: 队列中的任务并发执行。
- 主队列: 跟主线程相关的队列,主队列里面的内容都会在主线程中执行(我们一般在主线程中刷新UI)。
- 全局队列: 一个特殊的并发队列。
-
并发队列与全局队列的区别:
- 并发队列有名称,可以跟踪错误。全局队列没有
- 在ARC中两个队列不需要考虑释放内存,但是在MRC中并发队列是创建出来的需要release操作,而全局队列只有一个不需要。
- 一般在开发过程中我们使用全局队列。
-
执行任务的两个函数
- '同步'执行任务:dispatch_sync(<#dispatch_queue_t queue#>, <#^(void)block#>)
- '异步'执行任务:dispatch_async(dispatch_queue_t queue, <#^(void)block#>)
-
"同步"和"异步"的区别:
- "同步": 只能在'当前'线程中执行任务,不具备开启新线程的能力.
- "异步": 可以在'新'的线程中执行任务,具备开启新线程的能力.
-
各个队列的执行效果:
- 串行队列同步执行,既在当前线程中顺序执行
- 串行队列异步执行,开辟一条新的线程,在该线程中顺序执行
- 并行队列同步执行,不开辟线程,在当前线程中顺序执行
- 并行队列异步执行,开辟多个新的线程,并且线程会重用,无序执行
- 主队列异步执行,不开辟新的线程,顺序执行
- 主队列同步执行,会造成死锁('主线程'和'主队列'相互等待,卡住主线程)
-
线程间通讯:经典案例:子线程进行耗时操作(例如下载更新等)主线程进行UI刷新。
- 经典用法(子线程下载(耗时操作),主线程刷新UI):
dispatch_async(dispatch_get_global_queue(0, 0), ^{
// 执行耗时的异步操作...
dispatch_async(dispatch_get_main_queue(), ^{
// 回到主线程,执行UI刷新操作
-
延迟操作
调用 NSObject 方法:[self performSelector:@selector(run) withObject:nil afterDelay:2.0];
// 2秒后再调用self的run方法
GCD函数实现延时执行:dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
// 2秒后执行这里的代码... 在哪个线程执行,跟队列类型有关
-
队列组的使用:
- 项目需求:首先:分别异步执行两个耗时操作;其次:等两次耗时操作都执行完毕后,再回到主线程执行操作.使用队列组(dispatch_group_t)快速,高效的实现上述需求.
dispatch_group_t group = dispatch_group_create(); // 队列组
dispatch_queue_t queue = dispatch_get_global_queue(0, 0); // 全局并发队列
dispatch_group_async(group, queue, ^{// 异步执行操作1
// longTime1
});
dispatch_group_async(group, queue, ^{ // 异步执行操作2
// longTime2
});
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
// 在主线程刷新数据
// reload Data
});
GCD内部怎么实现的
- iOS和OS X的核心是XNU内核,GCD是基于XNU内核实现的
- GCD的API全部在libdispatch库中
- GCD的底层实现主要有Dispatch Queue和Dispatch Source
- Dispatch Queue :管理block(操作)
- Dispatch Source :处理事件(MACH端口发送,MACH端口接收,检测与进程相关事件等10种事件)
GCD的queue、main queue中执行的代码一定是在main thread么?
? 对于queue中所执行的代码不一定在main thread中。如果queue是在主线程中创建的,那么所执行的代码就是在主线程中执行。如果是在子线程中创建的,那么就不会在main thread中执行。
? 对于main queue就是在主线程中的,因此一定会在主线程中执行?;袢ain queue就可以了,不需要我们创建,获取方式通过调用方法dispatchgetmain_queue来获取。
如何用GCD同步若干个异步调用?(如根据若干个url异步加载多张图片,然后在都下载完成后合成一张整图)
使用Dispatch Group追加block到Global Group Queue,这些block如果全部执行完毕,就会执行Main Dispatch Queue中的结束处理的block。
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, );
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{ /*加载图片1 */ });
dispatch_group_async(group, queue, ^{ /*加载图片2 */ });
dispatch_group_async(group, queue, ^{ /*加载图片3 */ });
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
// 合并图片
});
有a、b、c、d 4个异步请求,如何判断a、b、c、d都完成执行?如果需要a、b、c、d顺序执行,该如何实现?
1. 对于这四个异步请求,要判断都执行完成最简单的方式就是通过GCD的group来实现:
2. dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
3. dispatch_group_t group = dispatch_group_create();
4. dispatch_group_async(group, queue, ^{ /*任务a */ });
5. dispatch_group_async(group, queue, ^{ /*任务b */ });
6. dispatch_group_async(group, queue, ^{ /*任务c */ });
7. dispatch_group_async(group, queue, ^{ /*任务d */ });
8.
9. dispatch_group_notify(group, dispatch_get_main_queue(), ^{
10. // 在a、b、c、d异步执行完成后,会回调这里
11. });
当然,我们还可以使用非常老套的方法来处理,通过四个变量来标识a、b、c、d四个任务是否完成,然后在runloop中让其等待,当完成时才退出run loop。但是这样做会让后面的代码得不到执行,直到Run loop执行完毕。
要求顺序执行,那么可以将任务放到串行队列中,自然就是按顺序来异步执行了
发送10个网络请求,然后再接收到所有回应之后执行后续操作,如何实现?
从题目分析可知,10个请求要全部完成后,才执行某一功能。比如,下载10图片后合成一张大图,就需要异步全部下载完成后,才能合并成大图。
做法:通过dispatch_group_t来实现,将每个请求放入到Group中,将合并成大图的操作放在dispatch_group_notify中实现。
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{ /*加载图片1 */ });
dispatch_group_async(group, queue, ^{ /*加载图片2 */ });
dispatch_group_async(group, queue, ^{ /*加载图片3 */ });
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
// 合并图片
});
苹果为什么要废弃dispatch_get_current_queue?
- dispatch_get_current_queue容易造成死锁。详情点击该API查看官方注释。
如果让你来实现 dispatch_once,你会怎么做?
http://www.dreamingwish.com/article/gcd-guide-dispatch-once-2.html(超级详细解析)
个人觉得说出实现的思路即可,无锁的线程同步编程,每一处的线程竞争都考虑到并妥善处理
线程A执行Block时,任何其它线程都需要等待。
线程A执行完Block应该立即标记任务完成状态,然后遍历信号量链来唤醒所有等待线程。
线程A遍历信号量链来signal时,任何其他新进入函数的线程都应该直接返回而无需等待。
线程A遍历信号量链来signal时,若有其它等待线程B仍在更新或试图更新信号量链,应该保证此线程B能正确完成其任务:a.直接返回 b.等待在信号量上并很快又被唤醒。
线程B构造信号量时,应该考虑线程A随时可能改变状态(“等待”、“完成”、“遍历信号量链”)。
线程B构造信号量时,应该考虑到另一个线程C也可能正在更新或试图更新信号量链,应该保证B、C都能正常完成其任务:a.增加链节并等待在信号量上 b.发现线程A已经标记“完成”然后直接销毁信号量并退出函数。
关于NSOperation:
- NSOperation: 抽象类,不能直接使用,需要使用其子类.(类似的类还有核心动画)
- 两个常用子类: NSInvocationOperation(调用) 和 NSBlockOperation(块);
- 两者没有本质区别,后者使用 Block 的形式组织代码,使用相对方便.
- NSInvocationOperation在调用start方法后,不会开启新的线程只会在当前线程中执行。
- NSBlockOperation 在调用start方法后,如果封装的操作数>1会开辟多条线程执行 =1 只会在当前线程执行.
- NSOperationQueue 创建的操作队列默认为全局队列,队列中的操作执行顺序是无序的,如果需要让它有序执行需要添加依赖关系。
- // 操作op3依赖于操作op2 [op3 addDependency:op2];
- // 操作op2依赖于操作op1 [op2 addDependency:op1];
- 同时可以设置最大并发数
- NSOperationQueue NSOperation支持 取消暂停的操作 但是正在进行的的操作并不能取消,这一旦取消不可恢复.
- NSOperationQueue支持KVO,可以监测operation是否正在执行(isExecuted)、是否结束(isFinished),是否取消(isCanceld)
NSOperation queue?
- 存放NSOperation的集合类。不能说队列,不是严格的先进先出
NSOperation与GCD的区别
-
GCD
- GCD是iOS4.0推出的,主要针对多核cpu做了优化,是纯c语言的技术。
- GCD是将任务(block)添加到队列(串行、并行、全局、主队列),并且以同步/异步的方式执行任务的函数。
- GCD提供了一些NSOperation不具备的功能
- 一次性执行
- 延迟执行
- 调度组
- GCD 是严格的队列,先进先出 FIFO;
-
NSOperation
- NSOperation是iOS2.0推出的,iOS4.0以后又重写了NSOperation
- NSOperation是将操作(异步的任务)添加到队列(并发队列),就会执行指定的函数
- NSOperation提供的方便操作
- 最大并发数
- 队列的暂停和继续
- 取消所有的操作
- 指定操作之间的依赖关系依赖关系,可以让异步任务同步执行.
- 将KVO用于NSOperation中,监听一个operation是否完成。
- 能够设置NSOperation的优先级,能够使同一个并行队列中的任务区分先后地执行
- 对NSOperation进行继承,在这之上添加成员变量与成员方法,提高整个代码的复用度
GCD与NSThread的区别
- NSThread 通过 @selector 指定要执行的方法,代码分散
- GCD 通过 block 指定要执行的代码,代码集中, 所有的代码写在一起的,让代码更加简单,易于阅读和维护
- 使用 GCD 不需要管理线程的创建/销毁/复用的过程!程序员不用关心线程的生命周期
- 如果要开多个线程 NSThread 必须实例化多个线程对象
- NSThread 靠 NSObject 的分类方法实现的线程间通讯,
为什么要取消/恢复队列呢?
- 一般在内存警告后取消队列中的操作。
- 为了保证scorllView在滚动的时候流畅 通常在滚动开始时,暂停队列中的所有操作,滚动结束后,恢复操作。
Object C中创建线程的方法是什么?如果在主线程中执行代码,方法是什么?如果想延时执行代码、方法又是什么?
线程创建有三种方法:使用NSThread创建、使用GCD的dispatch、使用子类化的NSOperation,然后将其加入NSOperationQueue;
NSThread创建线程的三种方法:
NSThread *thread = [[NSThread alloc]initWithTarget:self selector:@selector(run:) object:@"nil"];
[NSThread detachNewThreadSelector:@selector(run:) toTarget:self withObject:@"我是分离出来的子线程"];
[self performSelectorInBackground:@selector(run:) withObject:@"我是后台线程"];
在主线程执行代码,就调用performSelectorOnMainThread方法。
如果想延时执行代码可以调用performSelector:onThread:withObject:waitUntilDone:方法;
GCD:
利用异步函数dispatch_async()创建子线程。
在主线程执行代码,dispatch_async(dispatch_get_main_queue(), ^{});
延迟执行代码(延迟·可以控制代码在哪个线程执行):
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{});
NSOperationQueue:
使用NSOperation的子类封装操作,再将操作添加到NSOperationQueue创建的队列中,实现多线程。
在主线程执行代码,只要将封装代码的NSOperation对象添加到主队列就可以了。
下面关于线程管理错误的是
A. GCD所用的开销要比NSThread大
B. 可以在子线程中修改UI元素
C. NSOperationQueue是比NSthread更高层的封装
D. GCD可以根据不同优先级分配线程
- 参考答案:B
- 理由:首先,UI元素的更新必须在主线程。GCD与Block配合使用,block需要自动捕获上下文变量信息等,因此需要更多的资源,故比NSThread开销要大一些。NSOperationQueue与NSOperation配合使用,比NSThread更易于操作线程。GCD提供了多个优先级,我们可以根据设置优先级,让其自动为我们分配线程。
文章如有问题,请留言,我将及时更正。
满地打滚卖萌求赞,如果本文帮助到你,轻点下方的红心,给作者君增加更新的动力。