并發(fā)場(chǎng)景下死鎖

案例場(chǎng)景

例如賬戶A 轉(zhuǎn)賬戶B琼娘、賬戶C 轉(zhuǎn)賬戶D這兩個(gè)轉(zhuǎn)賬操作。

class Account {
  private int balance;
  // 轉(zhuǎn)賬
  void transfer(Account target, int amt){
    // 鎖定轉(zhuǎn)出賬戶
    synchronized(this){     ①
      // 鎖定轉(zhuǎn)入賬戶
      synchronized(target){ ②
        if (this.balance > amt) {
          this.balance -= amt;
          target.balance += amt;
        }
      }
    }
  } 
}

這種方式采用了細(xì)粒度鎖赁濒。使用細(xì)粒度鎖可以提高并行度轨奄,是性能優(yōu)化的一個(gè)重要手段。

但是會(huì)出現(xiàn)一個(gè)新的問題就是死鎖拒炎。

上面轉(zhuǎn)賬的代碼是怎么發(fā)生死鎖的呢挪拟?我們假設(shè)線程T1執(zhí)行賬戶A轉(zhuǎn)賬戶B的操作,賬戶A.transfer(賬戶B)击你;同時(shí)線程T2執(zhí)行賬戶B轉(zhuǎn)賬戶A的操作玉组,賬戶B.transfer(賬戶A)。當(dāng)T1和T2同時(shí)執(zhí)行完①處的代碼時(shí)丁侄,T1獲得了賬戶A的鎖(對(duì)于T1惯雳,this是賬戶A),而T2獲得了賬戶B的鎖(對(duì)于T2鸿摇,this是賬戶B)石景。之后T1和T2在執(zhí)行②處的代碼時(shí),T1試圖獲取賬戶B的鎖時(shí)拙吉,發(fā)現(xiàn)賬戶B已經(jīng)被鎖定(被T2鎖定)潮孽,所以T1開始等待;T2則試圖獲取賬戶A的鎖時(shí)筷黔,發(fā)現(xiàn)賬戶A已經(jīng)被鎖定(被T1鎖定)往史,所以T2也開始等待。于是T1和T2會(huì)無(wú)期限地等待下去佛舱,也就是我們所說的死鎖了椎例。

如何預(yù)防死鎖

產(chǎn)生死鎖的四個(gè)必要條件:

  • 互斥,共享資源X和Y只能被一個(gè)線程占用请祖;
  • 占有且等待订歪,線程T1已經(jīng)取得共享資源X,在等待共享資源Y的時(shí)候损拢,不釋放共享資源X陌粹;
  • 不可搶占,其他線程不能強(qiáng)行搶占線程T1占有的資源;
  • 循環(huán)等待掏秩,線程T1等待線程T2占有的資源或舞,線程T2等待線程T1占有的資源,就是循環(huán)等待蒙幻。

其中映凳,互斥這個(gè)條件我們沒有辦法破壞,因?yàn)槲覀冇面i為的就是互斥邮破。不過其他三個(gè)條件都是有辦法破壞掉的诈豌,到底如何做呢?

  • 破壞占用且等待條件

可以一次性申請(qǐng)所有資源抒和,新增一個(gè)賬本管理員的概念矫渔,兩個(gè)資源只能同時(shí)給一個(gè)線程。這個(gè)管理員只能是一個(gè)單例摧莽,因?yàn)橹挥幸粋€(gè)管理員庙洼。

class Allocator {
  private List<Object> als =
    new ArrayList<>();
  // 一次性申請(qǐng)所有資源
  synchronized boolean apply(
    Object from, Object to){
    if(als.contains(from) ||
         als.contains(to)){
      return false;  
    } else {
      als.add(from);
      als.add(to);  
    }
    return true;
  }
  // 歸還資源
  synchronized void free(
    Object from, Object to){
    als.remove(from);
    als.remove(to);
  }
}

class Account {
  // actr應(yīng)該為單例
  private Allocator actr;
  private int balance;
  // 轉(zhuǎn)賬
  void transfer(Account target, int amt){
    // 一次性申請(qǐng)轉(zhuǎn)出賬戶和轉(zhuǎn)入賬戶,直到成功
    while(!actr.apply(this, target))
      镊辕;
    try{
      // 鎖定轉(zhuǎn)出賬戶
      synchronized(this){              
        // 鎖定轉(zhuǎn)入賬戶
        synchronized(target){           
          if (this.balance > amt){
            this.balance -= amt;
            target.balance += amt;
          }
        }
      }
    } finally {
      actr.free(this, target)
    }
  } 
}
  • 破壞不可搶占條件

核心是要能夠主動(dòng)釋放它占有的資源油够,這一點(diǎn)synchronized是做不到的。原因是synchronized申請(qǐng)資源的時(shí)候征懈,如果申請(qǐng)不到石咬,線程直接進(jìn)入阻塞狀態(tài)了,而線程進(jìn)入阻塞狀態(tài)卖哎,啥都干不了鬼悠,也釋放不了線程已經(jīng)占有的資源。

lock可以后續(xù)補(bǔ)充亏娜。

  • 破壞循環(huán)等待條件

用“等待-通知”機(jī)制優(yōu)化循環(huán)等待
image

為什么說是曾經(jīng)滿足過呢厦章?因?yàn)閚otify()只能保證在通知時(shí)間點(diǎn),條件是滿足的照藻。而被通知線程的執(zhí)行時(shí)間點(diǎn)和通知的時(shí)間點(diǎn)基本上不會(huì)重合,所以當(dāng)線程執(zhí)行的時(shí)候汗侵,很可能條件已經(jīng)不滿足了(保不齊有其他線程插隊(duì))幸缕。被通知的線程要想重新執(zhí)行,仍然需要獲取到互斥鎖(因?yàn)樵?jīng)獲取的鎖在調(diào)用wait()時(shí)已經(jīng)釋放了)

notify()是會(huì)隨機(jī)地通知等待隊(duì)列中的一個(gè)線程晰韵,而notifyAll()會(huì)通知等待隊(duì)列中的所有線程发乔。

class Allocator {
  private List<Object> als;
  // 一次性申請(qǐng)所有資源
  synchronized void apply(
    Object from, Object to){
    // 經(jīng)典寫法
    while(als.contains(from) ||
         als.contains(to)){
      try{
        wait();
      }catch(Exception e){
      }   
    } 
    als.add(from);
    als.add(to);  
  }
  // 歸還資源
  synchronized void free(
    Object from, Object to){
    als.remove(from);
    als.remove(to);
    notifyAll();
  }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市雪猪,隨后出現(xiàn)的幾起案子栏尚,更是在濱河造成了極大的恐慌,老刑警劉巖只恨,帶你破解...
    沈念sama閱讀 219,589評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件译仗,死亡現(xiàn)場(chǎng)離奇詭異抬虽,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)纵菌,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,615評(píng)論 3 396
  • 文/潘曉璐 我一進(jìn)店門阐污,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人咱圆,你說我怎么就攤上這事笛辟。” “怎么了序苏?”我有些...
    開封第一講書人閱讀 165,933評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵手幢,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我忱详,道長(zhǎng)围来,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,976評(píng)論 1 295
  • 正文 為了忘掉前任踱阿,我火速辦了婚禮管钳,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘软舌。我一直安慰自己才漆,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,999評(píng)論 6 393
  • 文/花漫 我一把揭開白布佛点。 她就那樣靜靜地躺著醇滥,像睡著了一般。 火紅的嫁衣襯著肌膚如雪超营。 梳的紋絲不亂的頭發(fā)上鸳玩,一...
    開封第一講書人閱讀 51,775評(píng)論 1 307
  • 那天,我揣著相機(jī)與錄音演闭,去河邊找鬼不跟。 笑死,一個(gè)胖子當(dāng)著我的面吹牛米碰,可吹牛的內(nèi)容都是我干的窝革。 我是一名探鬼主播,決...
    沈念sama閱讀 40,474評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼吕座,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼虐译!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起吴趴,我...
    開封第一講書人閱讀 39,359評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤漆诽,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體厢拭,經(jīng)...
    沈念sama閱讀 45,854評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡兰英,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,007評(píng)論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了蚪腐。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片箭昵。...
    茶點(diǎn)故事閱讀 40,146評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖回季,靈堂內(nèi)的尸體忽然破棺而出家制,到底是詐尸還是另有隱情,我是刑警寧澤泡一,帶...
    沈念sama閱讀 35,826評(píng)論 5 346
  • 正文 年R本政府宣布颤殴,位于F島的核電站,受9級(jí)特大地震影響鼻忠,放射性物質(zhì)發(fā)生泄漏涵但。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,484評(píng)論 3 331
  • 文/蒙蒙 一帖蔓、第九天 我趴在偏房一處隱蔽的房頂上張望矮瘟。 院中可真熱鬧,春花似錦塑娇、人聲如沸澈侠。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,029評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)哨啃。三九已至,卻和暖如春写妥,著一層夾襖步出監(jiān)牢的瞬間拳球,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,153評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工珍特, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留祝峻,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,420評(píng)論 3 373
  • 正文 我出身青樓扎筒,卻偏偏與公主長(zhǎng)得像呼猪,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子砸琅,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,107評(píng)論 2 356

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

  • Java多線程學(xué)習(xí) [-] 一擴(kuò)展javalangThread類 二實(shí)現(xiàn)javalangRunnable接口 三T...
    影馳閱讀 2,960評(píng)論 1 18
  • 本文主要講了java中多線程的使用方法、線程同步轴踱、線程數(shù)據(jù)傳遞症脂、線程狀態(tài)及相應(yīng)的一些線程函數(shù)用法、概述等。 首先講...
    李欣陽(yáng)閱讀 2,456評(píng)論 1 15
  • 1. cpu通過時(shí)間片分配算法來(lái)循環(huán)執(zhí)行任務(wù)诱篷,當(dāng)前任務(wù)執(zhí)行一個(gè)時(shí)間片后會(huì)切換到下一任務(wù)壶唤。但是,再切換之前會(huì)保存上一...
    冰與河豚魚閱讀 669評(píng)論 0 0
  • 1棕所、競(jìng)態(tài)條件: 定義:競(jìng)態(tài)條件指的是一種特殊的情況闸盔,在這種情況下各個(gè)執(zhí)行單元以一種沒有邏輯的順序執(zhí)行動(dòng)作,從而導(dǎo)致...
    Hughman閱讀 1,293評(píng)論 0 7
  • 一擴(kuò)展javalangThread類二實(shí)現(xiàn)javalangRunnable接口三Thread和Runnable的區(qū)...
    和帥_db6a閱讀 487評(píng)論 0 1