552,GCD的死鎖(面試點(diǎn):gcd死鎖:一末誓,主線程調(diào)用主線程。二书蚪,同步串行隊(duì)列嵌套自己喇澡。三,信號(hào)量阻塞主線程)

本文主要舉例說明GCD里的死鎖場(chǎng)景殊校,分析造成死鎖的原因以及解決方案

在開始說GCD死鎖之前晴玖,我們先了解一下GCD的中的任務(wù)派發(fā)和隊(duì)列。

任務(wù)派發(fā)

任務(wù)派發(fā)方式 說明
dispatch_sync() 同步執(zhí)行为流,完成了它預(yù)定的任務(wù)后才返回呕屎,阻塞當(dāng)前線程
dispatch_async() 異步執(zhí)行,會(huì)立即返回敬察,預(yù)定的任務(wù)會(huì)完成但不會(huì)等它完成秀睛,不阻塞當(dāng)前線程

隊(duì)列種類

隊(duì)列種類 說明
串行隊(duì)列 同步執(zhí)行,完成了它預(yù)定的任務(wù)后才返回莲祸,阻塞當(dāng)前線程
并發(fā)隊(duì)列 一次可以并發(fā)執(zhí)行多個(gè)任務(wù)蹂安,不必等待執(zhí)行中的任務(wù)完成

GCD隊(duì)列種類

GCD隊(duì)列種類 獲取方法 隊(duì)列類型 說明
主隊(duì)列 dispatch_get_main_queue 串行隊(duì)列 主線中執(zhí)行
全局隊(duì)列 dispatch_get_global_queue 并發(fā)隊(duì)列 子線程中執(zhí)行
用戶隊(duì)列 dispatch_queue_create 串并都可以 子線程中執(zhí)行

GCD死鎖

在GCD中椭迎,主要的死鎖就是當(dāng)前串行隊(duì)列里面同步執(zhí)行當(dāng)前串行隊(duì)列。解決的方法就是將同步的串行隊(duì)列放到另外一個(gè)線程執(zhí)行田盈。

死鎖場(chǎng)景

1. 死鎖場(chǎng)景: 主線程調(diào)用主線程

- (void)deadLockCase1 {
    NSLog(@"1"); // 任務(wù)1
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"2"); // 任務(wù)2
    });
    NSLog(@"3"); // 任務(wù)3
}

控制臺(tái)輸出:

1

原因:
從控制臺(tái)輸出可以看出畜号,任務(wù)2和任務(wù)3沒有執(zhí)行,此時(shí)已經(jīng)死鎖了允瞧。
因?yàn)閐ispatch_sync是同步的简软,本身就會(huì)阻塞當(dāng)前線程,此刻阻塞了主線程瓷式。而當(dāng)前block又在等待主線程執(zhí)行完畢替饿,從而形成了主線程等待主線程语泽,自己等自己的情況贸典,形成了死鎖。

解決方法:
1.改用異步dispatch_async執(zhí)行

NSLog(@"1"); // 任務(wù)1
dispatch_async(dispatch_get_main_queue(), ^{
        NSLog(@"2"); // 任務(wù)2
 });
NSLog(@"3"); // 任務(wù)3

控制臺(tái)輸出:

1
3
2 

2.不在主線程中運(yùn)行踱卵,而是放在子線程中

NSLog(@"1"); // 任務(wù)1
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{
        NSLog(@"2"); // 任務(wù)2
});
NSLog(@"3"); // 任務(wù)3

控制臺(tái)輸出:

1
2
3

:如果block中是刷新UI的操作廊驼,則不能放在子線程中執(zhí)行,會(huì)crash

死鎖場(chǎng)景2: (同步串行隊(duì)列嵌套自己)

- (void)deadLockCase2 {
    dispatch_queue_t aSerialDispatchQueue = dispatch_queue_create("com.test.deadlock.queue", DISPATCH_QUEUE_SERIAL);
    NSLog(@"1"); //任務(wù)1
    dispatch_sync(aSerialDispatchQueue, ^{
        NSLog(@"2"); //任務(wù)2
        dispatch_sync(aSerialDispatchQueue, ^{
            NSLog(@"3"); //任務(wù)3
        });
        NSLog(@"4");  //任務(wù)4
    });
    NSLog(@"5");  //任務(wù)5
}

控制臺(tái)輸出:

1
2

原因:

從控制臺(tái)輸出結(jié)果來看惋砂,執(zhí)行到任務(wù)2后妒挎,就已經(jīng)死鎖了。因?yàn)樵摾又袃蓚€(gè)GCD都是使用的同步方式西饵,而且還是同一個(gè)串行隊(duì)列酝掩,這就導(dǎo)致了和上一個(gè)例子一樣,自己在等待自己的情況眷柔,形成了死鎖期虾。

解決方法:

  1. 將第二個(gè)GCD改為異步
 - (void)deadLockCase2 {
    dispatch_queue_t aSerialDispatchQueue = dispatch_queue_create("com.test.deadlock.queue", DISPATCH_QUEUE_SERIAL);
    NSLog(@"1"); //任務(wù)1
    dispatch_sync(aSerialDispatchQueue, ^{
        NSLog(@"2"); //任務(wù)2
        dispatch_async(aSerialDispatchQueue, ^{
            NSLog(@"3"); //任務(wù)3
        });
        NSLog(@"4");  //任務(wù)4
    });
    NSLog(@"5");  //任務(wù)5
}

控制臺(tái)輸出:

1
2
4
5
3

然而,將第一個(gè)GCD改為異步驯嘱,不能解決問題

 - (void)deadLockCase2 {
    dispatch_queue_t aSerialDispatchQueue = dispatch_queue_create("com.test.deadlock.queue", DISPATCH_QUEUE_SERIAL);
    NSLog(@"1"); //任務(wù)1
    dispatch_async(aSerialDispatchQueue, ^{
        NSLog(@"2"); //任務(wù)2
        dispatch_sync(aSerialDispatchQueue, ^{
            NSLog(@"3"); //任務(wù)3
        });
        NSLog(@"4");  //任務(wù)4
    });
    NSLog(@"5");  //任務(wù)5
}

控制臺(tái)輸出:

1
5
2

原因:

雖然第一個(gè)GCD是異步的镶苞,但是第二個(gè)GCD是同步的,第二個(gè)GCD在等著第一個(gè)GCD結(jié)束鞠评,而第一個(gè)GCD的block又在等著第一個(gè)GCD結(jié)束茂蚓,這樣就形成了死鎖。
注:對(duì)于以上將第二個(gè)GCD改為異步剃幌,第一個(gè)GCD為同步的場(chǎng)景聋涨,不會(huì)造成死鎖,是因?yàn)榈诙€(gè)GCD為異步负乡,它不用等待第一個(gè)GCD執(zhí)行完畢牍白,它和第一個(gè)GCD是沒有同步關(guān)系的。它是在第一個(gè)GCD執(zhí)行的同時(shí)并發(fā)執(zhí)行自己block的代碼敬鬓。

2.將兩個(gè)GCD都改為異步

 - (void)deadLockCase2 {
    dispatch_queue_t aSerialDispatchQueue = dispatch_queue_create("com.test.deadlock.queue", DISPATCH_QUEUE_SERIAL);
    NSLog(@"1"); //任務(wù)1
    dispatch_async(aSerialDispatchQueue, ^{
        NSLog(@"2"); //任務(wù)2
        dispatch_async(aSerialDispatchQueue, ^{
            NSLog(@"3"); //任務(wù)3
        });
        NSLog(@"4");  //任務(wù)4
    });
    NSLog(@"5");  //任務(wù)5
}

控制臺(tái)輸出:

1
5
2
4
3
  1. 使用不同的串行隊(duì)列
 - (void)deadLockCase2 {
    dispatch_queue_t aSerialDispatchQueue1 = dispatch_queue_create("com.test.deadlock.queue1", DISPATCH_QUEUE_SERIAL);
    dispatch_queue_t aSerialDispatchQueue2 = dispatch_queue_create("com.test.deadlock.queue2", DISPATCH_QUEUE_SERIAL);
    NSLog(@"1"); //任務(wù)1
    dispatch_sync(aSerialDispatchQueue1, ^{
        NSLog(@"2"); //任務(wù)2
        dispatch_sync(aSerialDispatchQueue2, ^{
            NSLog(@"3"); //任務(wù)3
        });
        NSLog(@"4");  //任務(wù)4
    });
    NSLog(@"5");  //任務(wù)5
}

控制臺(tái)輸出:

1
2
3
4
5

3.死鎖場(chǎng)景: 信號(hào)量阻塞主線程

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
    NSLog(@"semaphore create!");
    dispatch_async(dispatch_get_main_queue(), ^{
        dispatch_semaphore_signal(semaphore);
        NSLog(@"semaphore plus 1");
    });
    dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    NSLog(@"semaphore minus 1");
}
原因:

如果當(dāng)前執(zhí)行的線程是主線程淹朋,以上代碼就會(huì)出現(xiàn)死鎖笙各。
因?yàn)?code>dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER)阻塞了當(dāng)前線程,而且等待時(shí)間是DISPATCH_TIME_FOREVER——永遠(yuǎn)等待础芍,這樣它就永遠(yuǎn)的阻塞了當(dāng)前線程——主線程杈抢。導(dǎo)致主線中的dispatch_semaphore_signal(semaphore)沒有執(zhí)行,
dispatch_semaphore_wait一直在等待dispatch_semaphore_signal改變信號(hào)量仑性,這樣就形成了死鎖惶楼。

解決方法:

應(yīng)該將信號(hào)量移到并行隊(duì)列中,如全局調(diào)度隊(duì)列诊杆。以下場(chǎng)景歼捐,移到串行隊(duì)列也是可以的。但是串行隊(duì)列還是有可能死鎖的(如果執(zhí)行dispatch_semaphore_signal方法還是在對(duì)應(yīng)串行隊(duì)列中的話晨汹,即之前提到的串行隊(duì)列嵌套串行隊(duì)列的場(chǎng)景)豹储。

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
        NSLog(@"semaphore create!");
        dispatch_async(dispatch_get_main_queue(), ^{
            dispatch_semaphore_signal(semaphore);
            NSLog(@"semaphore plus 1");
        });
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
        NSLog(@"semaphore minus 1");
    });
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市淘这,隨后出現(xiàn)的幾起案子剥扣,更是在濱河造成了極大的恐慌,老刑警劉巖铝穷,帶你破解...
    沈念sama閱讀 212,454評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件钠怯,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡曙聂,警方通過查閱死者的電腦和手機(jī)晦炊,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來宁脊,“玉大人断国,你說我怎么就攤上這事‰澹” “怎么了并思?”我有些...
    開封第一講書人閱讀 157,921評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長语稠。 經(jīng)常有香客問我宋彼,道長,這世上最難降的妖魔是什么仙畦? 我笑而不...
    開封第一講書人閱讀 56,648評(píng)論 1 284
  • 正文 為了忘掉前任输涕,我火速辦了婚禮,結(jié)果婚禮上慨畸,老公的妹妹穿的比我還像新娘莱坎。我一直安慰自己,他們只是感情好寸士,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,770評(píng)論 6 386
  • 文/花漫 我一把揭開白布檐什。 她就那樣靜靜地躺著碴卧,像睡著了一般。 火紅的嫁衣襯著肌膚如雪乃正。 梳的紋絲不亂的頭發(fā)上住册,一...
    開封第一講書人閱讀 49,950評(píng)論 1 291
  • 那天,我揣著相機(jī)與錄音瓮具,去河邊找鬼荧飞。 笑死,一個(gè)胖子當(dāng)著我的面吹牛名党,可吹牛的內(nèi)容都是我干的叹阔。 我是一名探鬼主播,決...
    沈念sama閱讀 39,090評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼传睹,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼耳幢!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起蒋歌,我...
    開封第一講書人閱讀 37,817評(píng)論 0 268
  • 序言:老撾萬榮一對(duì)情侶失蹤帅掘,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后堂油,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,275評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡碧绞,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,592評(píng)論 2 327
  • 正文 我和宋清朗相戀三年府框,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片讥邻。...
    茶點(diǎn)故事閱讀 38,724評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡迫靖,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出兴使,到底是詐尸還是另有隱情系宜,我是刑警寧澤,帶...
    沈念sama閱讀 34,409評(píng)論 4 333
  • 正文 年R本政府宣布发魄,位于F島的核電站盹牧,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏励幼。R本人自食惡果不足惜汰寓,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,052評(píng)論 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望苹粟。 院中可真熱鬧有滑,春花似錦、人聲如沸嵌削。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至肌访,卻和暖如春荣茫,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背场靴。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評(píng)論 1 266
  • 我被黑心中介騙來泰國打工啡莉, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人旨剥。 一個(gè)月前我還...
    沈念sama閱讀 46,503評(píng)論 2 361
  • 正文 我出身青樓咧欣,卻偏偏與公主長得像,于是被迫代替她去往敵國和親轨帜。 傳聞我的和親對(duì)象是個(gè)殘疾皇子魄咕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,627評(píng)論 2 350

推薦閱讀更多精彩內(nèi)容