如何安全使用dispatch_sync
概述
iOS开发者在与线程打交道的方式中,使用最多的应该就是GCD框架了,没有之一。GCD将繁琐的线程抽象为了一个个队列,让开发者极易理解和使用。但其实队列的底层,依然是利用线程实现的,同样会有死锁的问题。本文将探讨如何规避disptach_sync接口引入的死锁问题。
GCD基础
GCD最基础的两个接口
dispatch_sync(dispatch_queue_t queue, dispatch_block_t block);
dispatch_async(dispatch_queue_t queue, dispatch_block_t block);
第一个参数queue为队列对象,第二个参数block为block对象。这两个接口可以将任务block扔到队列queue中去执行。
开发者使用最频繁的,就是在子线程环境下,需要做UI更新时,我们可以将任务扔到主线程去执行,
dispatch_sync(dispatch_get_main_queue(), block);
dispatch_async(dispatch_get_main_queue(), block);
而dispatch_sync(dispatch_get_main_queue(), block)有可能引入死锁的问题。
async VS.sync
disptach_async是异步扔一个block到queue中,即扔完我就不管了,继续执行我的下一行代码。实际上当下一行代码执行时,这个block还未执行,只是入了队列queue,queue会排队来执行这个block。
而disptach_sync则是同步扔一个block到queue中,即扔了我就等着,等到queue排队把这个block执行完了之后,才继续执行下一行代码。
为什么要使用sync
disptach_sync主要用于代码上下文对时序有强要求的场景。简单点说,就是下一行代码的执行,依赖于上一行代码的结果。例如说,我们需要在子线程中读取一个image对象,使用接口[UIImage imageNamed:],但imageNamed:实际上在iOS9以后才是线程安全的,iOS9之前都需要在主线程获取。所以,我们需要从子线程切换到主线程获取image,然后再切回子线程拿到这个image,
// ...currently in a subthread
__block UIImage *image;
dispatch_sync_on_main_queue(^{
image = [UIImage imageNamed:@"Resource/img"];
});
attachment.image = image;
这里我们必须使用sync。
为什么会死锁
假设当前我们的代码正在queue0中执行。然后我们调用disptach_sync将一个任务block1扔到queue0中执行,
// ... currently in queue0 or queue0's corresponding thread.
dispatch_sync(queue0, block1);
这时,dispatch_sync将等待queue0排队执行完block1,然后才能继续执行下一行代码。But,当前代码执行的环境也是queue0。假设当前执行的任务为block0。也就是说,block0在执行到一半时,需要等到自己的下一个任务block1执行完,自己才能继续执行。而block1排队在后面,需要等block0执行完才能执行。这时死锁就产生了,block0和block1互相等待执行,当前线程就卡死在dispatch_sync这行代码处。
我们发现的卡死问题,一般都是主线程死锁。一种较为常见的情况是,本身就已经在主线程了,还同步向主线程扔了一个任务:
// ... currently in the main thread
dispatch_sync(dispatch_get_main_queue(), block);
安全方法
YYKit中提供了一个同步扔任务到主线程的安全方法:
/**
Submits a block for execution on a main queue and waits until the block completes.
*/
static inline void dispatch_sync_on_main_queue(void (^block)()) {
if (pthread_main_np()) {
block();
} else {
dispatch_sync(dispatch_get_main_queue(), block);
}
}
其方式就是在扔任务给主线程之前,先检查当前线程是否已经是主线程,如果是,就不用调用GCD的队列调度接口dispatch_sync了,直接执行即可;如果不是主线程,那么调用GCD的dispatch_sync也不会卡死。
但事实上并不是这样的,dispatch_sync_on_main_queue也可能会卡死,这个安全接口并不安全。这个接口只能保证两个block之间不因互相等待而死锁。多于两个block的互相依赖就束手无策了。
举个例子,假设queue0是一个子线程的队列:
/* block0 */
// ... currently in the main thread.
dispatch_sync(queue0, ^{
/* block1 */
// ... currently in queue0's corresponding subthread.
dispatch_sync_on_main_queue(^{
/* block2 */
});
});
在上述代码中,block0正在主线程中执行,并且同步等待子线程执行完block1。block1又同步等待主线程执行完block2。而当前主线程正在执行block0,即block2的执行需要等到block0执行完。这样就成了block0-->block1-->block2-->block0...这样一个循环等待,即死锁。由于block1的环境是子线程,所以安全API的线程判断不起任何作用。
另举一个例子:
/* block0 */
// ... currently in the main thread.
[[NSNotificationCenter defaultCenter] postNotificationName:@"aNotification" object:nil];
// ... in another context
[[NSNotificationCenter defaultCenter] addObserverForName:@"aNotification"
object:nil
queue:queue0
usingBlock:^(NSNotification * _Nonnull note) {
/* block1 */
// ... currently in queue0's corresponding subthread.
dispatch_sync_on_main_queue(^{
/* block2 */
});
}];
由于通知NSNotification的执行是同步的,这里会出现和上一例一样的死锁情况:block0-->block1-->block2-->block0...
如何定位死锁问题
1.死锁监测和堆栈上报机制
要定位死锁的问题,我们需要知道在哪一行代码上死锁了,以及为什么会出现死锁。通常只要知道哪一行代码死锁了,我们就能通过代码分析出问题所在了。所以,如果死锁的时候,我们能够把堆栈上报上来,就能知道哪一行代码死锁了。这里需要有完善的死锁监测和堆栈上报机制。
2.打印日志
如果暂时没有人力或者技术支撑你去搭建完善的死锁监测和堆栈上报机制,那么你可以做一件简单的事情以协助你定位问题,那就是打印日志。在dispatch_sync或者加锁之前,打印一条日志。这样在用户反馈问题,或者测试重现问题的时候,提取日志便可分析出卡死的代码处。
如何安全使用dispatch_sync
答案是,尽量不要使用。没有哪一个接口是可以保证绝对安全的。必须要使用dispatch_sync的时候,尽量使用dispatch_sync_on_main_queue这个API。
作者:Joey_Xu
链接:https://www.jianshu.com/p/b3227582037d
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
如何安全使用dispatch_sync的更多相关文章
- GCD中的dispatch_sync、dispatch_sync 分别与串行、并行队列组合执行小实验
平常开发中会经常用gcd做一下多线程任务,但一直没有对同步.异步任务在串行.并行队列的执行情况做个全面的认识,今天写了个demo跑了下,还是有些新发现的. 代码如下: - (void)touchesB ...
- dispatch_sync may result in dead-lock
以下代码会引起死锁 dispatch_block_t block = ^{ ; i < ; i++) { NSLog(@"dispatch_sync:%d", i); } } ...
- dispatch_async & dispatch_sync
Clear that! dispatch_async 是将block发送到指定线程去执行,当前线程不会等待,会继续向下执行. dispatch_sync 也是将block发送到指定的线程去执行,但是当 ...
- 完整详细的说明GCD列(一)dispatch_async;dispatch_sync;dispatch_async_f;dispatch_sync_f
为什么要写这个系列,由于百度了一下.我们正在寻找一个非常比较片面的Blog.抄来抄去,写作是很粗糙. 所以,我想写这个系列,尝试记录官方网站GCD强大的全功能的表达.为了方便他们,也方便他人,假设有发 ...
- dispatch_sync和dispatch_async的区别
dispatch_sync 线程同步.dispatch_async线程异步 比如 //同步 dispatch_sync(dispatch_get_global_queue(DISPATCH_QUEUE ...
- ios同步线程(dispatch_sync)保证代码在主线程中执行
- (BOOL)transitionToNextPhase { // 保证代码在主线程 if (![[NSThread currentThread] isMainThread]) { dispatch ...
- dispatch_async 和dispatch_sync
dispatch_sync(),同步添加操作.他是等待添加进队列里面的操作完成之后再继续执行. dispatch_queue_t concurrentQueue = dispatch_queue_cr ...
- GCD学习(六) dispatch_async 和dispatch_sync
dispatch_sync(),同步添加操作.他是等待添加进队列里面的操作完成之后再继续执行. dispatch_queue_t concurrentQueue = dispatch_queue_cr ...
- viewDidLoad dispatch_sync
- (void)viewDidLoad { [super viewDidLoad]; NSLog(@"1"); dispatch_sync(dispatch_get_main_qu ...
- iOS Dispatch_sync 阻塞线程的原因
大家的知道在主队列上使用dispatch_sync(), - (void)testSyncMainThread { dispatch_queue_t main = dispatch_get_main_ ...
随机推荐
- 暑假集训D18总结
考试 本来考试时以为能AK的,结果全是因为手贱啊= = T1 瞎XX贪心 我竟然当成了数学 还拍了半天以为是对的 T2 组合数学 太简单 半个小时直接A T3 最长上升(非下降?)子序列 考试25,加 ...
- [HDU3038]How Many Answers Are Wrong(并查集)
传送门 和某题类似,只不过奇偶换成了和. ——代码 #include <cstdio> #include <iostream> #define N 1000001 int n, ...
- percona-toolkit使用教程
1:慢日志查询 [root@test_dx modify]# wget percona.com/get/pt-query-digest [root@test_dx modify]# file pt-q ...
- 动态加入的HTML的自己主动渲染
这两天在写一个用EasyUI的前台,遇到动态向Layout加入HTML内容时没有自己主动渲染的问题.查了一下网上的资料后得以解决.详细例如以下: $("#content").htm ...
- poj 3468 A Simple Problem with Integers(线段树+区间更新+区间求和)
题目链接:id=3468http://">http://poj.org/problem? id=3468 A Simple Problem with Integers Time Lim ...
- LeetCode 28 Divide Two Integers
Divide two integers without using multiplication, division and mod operator. 思路:1.先将被除数和除数转化为long的非负 ...
- commons-fileupload上传文件(1)
近期,写一个上传图片的功能.于是用到commons-fileupload这个组件.提过form提交表单到后台(这里没实用到structs框架).在后台List pl = dfu.parseReques ...
- fixed和absolute
fixed是相对于浏览器窗口固定 absolute是相对于整体网页固定.(整体网页包括所有的内容,包含右侧滑动条滑动所能看到的内容)
- python统计ES存储空间占用的代码
import os from os.path import join, getsize def get_dir_size(dir, suffix_filter=None): size = 0L if ...
- 【BZOJ 1598】 牛跑步
[题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=1598 [算法] A*求k短路 [代码] #include<bits/stdc+ ...