概述

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是异步扔一个blockqueue中,即扔完我就不管了,继续执行我的下一行代码。实际上当下一行代码执行时,这个block还未执行,只是入了队列queuequeue会排队来执行这个block

disptach_sync则是同步扔一个blockqueue中,即扔了我就等着,等到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执行完才能执行。这时死锁就产生了,block0block1互相等待执行,当前线程就卡死在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正在主线程中执行,并且同步等待子线程执行完block1block1又同步等待主线程执行完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的更多相关文章

  1. GCD中的dispatch_sync、dispatch_sync 分别与串行、并行队列组合执行小实验

    平常开发中会经常用gcd做一下多线程任务,但一直没有对同步.异步任务在串行.并行队列的执行情况做个全面的认识,今天写了个demo跑了下,还是有些新发现的. 代码如下: - (void)touchesB ...

  2. dispatch_sync may result in dead-lock

    以下代码会引起死锁 dispatch_block_t block = ^{ ; i < ; i++) { NSLog(@"dispatch_sync:%d", i); } } ...

  3. dispatch_async & dispatch_sync

    Clear that! dispatch_async 是将block发送到指定线程去执行,当前线程不会等待,会继续向下执行. dispatch_sync 也是将block发送到指定的线程去执行,但是当 ...

  4. 完整详细的说明GCD列(一)dispatch_async;dispatch_sync;dispatch_async_f;dispatch_sync_f

    为什么要写这个系列,由于百度了一下.我们正在寻找一个非常比较片面的Blog.抄来抄去,写作是很粗糙. 所以,我想写这个系列,尝试记录官方网站GCD强大的全功能的表达.为了方便他们,也方便他人,假设有发 ...

  5. dispatch_sync和dispatch_async的区别

    dispatch_sync 线程同步.dispatch_async线程异步 比如 //同步 dispatch_sync(dispatch_get_global_queue(DISPATCH_QUEUE ...

  6. ios同步线程(dispatch_sync)保证代码在主线程中执行

    - (BOOL)transitionToNextPhase { // 保证代码在主线程 if (![[NSThread currentThread] isMainThread]) { dispatch ...

  7. dispatch_async 和dispatch_sync

    dispatch_sync(),同步添加操作.他是等待添加进队列里面的操作完成之后再继续执行. dispatch_queue_t concurrentQueue = dispatch_queue_cr ...

  8. GCD学习(六) dispatch_async 和dispatch_sync

    dispatch_sync(),同步添加操作.他是等待添加进队列里面的操作完成之后再继续执行. dispatch_queue_t concurrentQueue = dispatch_queue_cr ...

  9. viewDidLoad dispatch_sync

    - (void)viewDidLoad { [super viewDidLoad]; NSLog(@"1"); dispatch_sync(dispatch_get_main_qu ...

  10. iOS Dispatch_sync 阻塞线程的原因

    大家的知道在主队列上使用dispatch_sync(), - (void)testSyncMainThread { dispatch_queue_t main = dispatch_get_main_ ...

随机推荐

  1. 【codeforces 797C】Minimal string

    [题目链接]:http://codeforces.com/contest/797/problem/C [题意] 一开始,给你一个字符串s:两个空字符串t和u; 你有两种合法操作; 1.将s的开头字符加 ...

  2. 【codeforces 757D】Felicity's Big Secret Revealed

    [题目链接]:http://codeforces.com/problemset/problem/757/D [题意] 给你一个01串; 让你分割这个01串; 要求2切..n+1切; 对于每一种切法 所 ...

  3. [网络流24题#9] [cogs734] 方格取数 [网络流,最大流最小割]

    将网格分为两部分,方法是黑白染色,即判断(i+j)&1即可,分开后从白色格子向黑色格子连边,每个点需要四条(边界点可能更少),也就是每个格子周围的四个方向.之后将源点和汇点分别于黑白格子连边, ...

  4. 【DEBUG】不能将参数 1 从“CString”转换为“const char *”

    1.  在vc6.0下用CString str;num = atoi(str);就可以顺利取到num: 但是同样代码拿到vs2008就报错,error C2664: "atoi": ...

  5. 洛谷——P2676 超级书架

    https://www.luogu.org/problem/show?pid=2676#sub 题目描述 Farmer John最近为奶牛们的图书馆添置了一个巨大的书架,尽管它是如此的大,但它还是几乎 ...

  6. 菜鸟nginx源代码剖析数据结构篇(十) 自旋锁ngx_spinlock

    菜鸟nginx源代码剖析数据结构篇(十) 自旋锁ngx_spinlock Author:Echo Chen(陈斌) Email:chenb19870707@gmail.com Blog:Blog.cs ...

  7. 四大传值详解:属性传值,单例传值,代理传值,block传值

    一:属性传值 传值情景:从前一个页面向后一个页面传值 a.在后一个页面,根据传值类型和个数,写属性 b.在前一个页面, 为属性赋值 c.在后一个页面, 使用值 例如: 第一个视图: #import & ...

  8. AWS OpsWorks新增Amazon RDS支持

    AWS OpsWorks是一个应用管理服务. 你可以通过它把你的应用在一个 堆栈中定义成为不同层的集合.每一个堆栈提供了须要安装和配置的软件包信息,同一时候也能部署不论什么在OpsWorks层中定义的 ...

  9. CodeForces - 743D Chloe and pleasant prizes

    Chloe and pleasant prizes time limit per test 2 seconds memory limit per test 256 megabytes input st ...

  10. hdu 2177(威佐夫博奕+打表)

    取(2堆)石子游戏 Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...