死鎖(DeadLock)

所謂死鎖: 是指兩個(gè)或兩個(gè)以上的進(jìn)程在執(zhí)行過(guò)程中杨幼,由于競(jìng)爭(zhēng)資源或者由于彼此通信而造成的一種阻塞的現(xiàn)象,若無(wú)外力作用四瘫,它們都將無(wú)法推進(jìn)下去欲逃。此時(shí)稱(chēng)系統(tǒng)處于死鎖狀態(tài)或系統(tǒng)產(chǎn)生了死鎖,這些永遠(yuǎn)在互相等待的進(jìn)程稱(chēng)為死鎖進(jìn)程洗做。
百度百科
我們使用加鎖機(jī)制來(lái)確保線(xiàn)程安全,但如果過(guò)度使用加鎖,則可能導(dǎo)致鎖順序死鎖(Lock-Ordering Deadlock).同樣,我們使用線(xiàn)程池和信號(hào)量來(lái)限制對(duì)資源的使用,但這些被限制的行為可能會(huì)導(dǎo)致資源死鎖(Resource DeadLock).
Java應(yīng)用程序無(wú)法從死鎖中恢復(fù)過(guò)來(lái),因此在程序設(shè)計(jì)時(shí)要排除那些可能導(dǎo)致死鎖出現(xiàn)的條件
下面對(duì)死鎖的一些簡(jiǎn)單介紹

一,死鎖示例
1,鎖順序死鎖
  • 線(xiàn)程以不同的順序來(lái)獲得相同的鎖,那么就可能出現(xiàn)死鎖
  • 若所有線(xiàn)程以固定的順序來(lái)獲得鎖,那么在程序中就不會(huì)出現(xiàn)鎖順序死鎖的問(wèn)題

示例:

public class Test {
    public static void main(String[] args) {
        LeftRightDeadLock deadLock = new LeftRightDeadLock();
        LeftRightThread leftRightThread = new LeftRightThread(deadLock);
        RightLeftThread rightLeftThread = new RightLeftThread(deadLock);
        leftRightThread.start();
        rightLeftThread.start();
    }
}

class LeftRightThread extends Thread{
    private LeftRightDeadLock deadLock;
    public LeftRightThread(LeftRightDeadLock deadLock) {
        this.deadLock = deadLock;
    }
    
    @Override
    public void run() {
        while(true){
            deadLock.leftRight();
        }
    }
}

class RightLeftThread extends Thread{
    private LeftRightDeadLock deadLock;
    public RightLeftThread(LeftRightDeadLock deadLock) {
        this.deadLock = deadLock;
    }
    
    @Override
    public void run() {
        while(true){
            deadLock.rightLeft();
        }
    }
}

class LeftRightDeadLock{
    private Object leftLock = new Object();
    private Object rightLock = new Object();
    public void leftRight(){
        synchronized (leftLock) {
            synchronized (rightLock) {
                System.out.println(Thread.currentThread().getName()+"  leftRight");
            }
        }
    }
    
    public void rightLeft(){
        synchronized (rightLock) {
            synchronized (leftLock) {
                System.out.println(Thread.currentThread().getName()+"  rightLeft");
            }
        }
    }

通過(guò)讓線(xiàn)程獲取鎖的順序一致來(lái)避免死鎖

class LeftRightDeadLock{
    private Object leftLock = new Object();
    private Object rightLock = new Object();
    public void leftRight(){
        synchronized (leftLock) {
            synchronized (rightLock) {
                System.out.println(Thread.currentThread().getName()+"  leftRight");
            }
        }
    }
    
    public void rightLeft(){
        synchronized (leftLock) {
            synchronized (rightLock) {
                System.out.println(Thread.currentThread().getName()+"  rightLeft");
            }
        }
    }

2,在協(xié)作對(duì)象之間發(fā)生的死鎖
  • 如果在持有鎖時(shí)調(diào)用某個(gè)外部方法,那么將出現(xiàn)活躍性問(wèn)題.在這個(gè)外部方法中可能會(huì)獲取其他鎖(這可能會(huì)產(chǎn)生死鎖),或者阻塞時(shí)間過(guò)長(zhǎng),導(dǎo)致其他線(xiàn)程無(wú)法及時(shí)獲得當(dāng)前被持有的鎖
  • 解決辦法---開(kāi)放調(diào)用
    如果在調(diào)用某個(gè)方法時(shí)不需要持有鎖,那么這種調(diào)用被稱(chēng)為開(kāi)放調(diào)用(Open Call)
public class Test {
    
    public static void main(String[] args) throws InterruptedException {
        First first = null;
        Second second = null;
        first = new First();
        second = new Second();
        first.setSecond(second);
        second.setFirst(first);
        
        FirstThread firstThread = new FirstThread(first);
        SecondThread secondThread = new SecondThread(second);
        firstThread.start();
        Thread.sleep(1000);
        secondThread.start();
    }
}

class FirstThread extends Thread{
    First first;
    FirstThread(First first){
        this.first = first;
    }
    
    @Override
    public void run() {
        for(;;){
            first.getFirst();
        }
    }
}

class SecondThread extends Thread{
    Second second;
    SecondThread(Second second){
        this.second = second;
    }
    
    @Override
    public void run() {
        for(;;){
            second.getSecond();
        }
    }
}

class First {
    private Second second;

    public synchronized String getFirst() {
        second.setFirst(this);
        System.out.println(Thread.currentThread());
        return "first";
    }
    
    public synchronized void setSecond(Second second){
        this.second = second;
    }
}

class Second {
    private First first;

    public synchronized String getSecond() {
        first.setSecond(this);
        System.out.println(Thread.currentThread());
        return "second";
    }
    
    public synchronized void setFirst(First first){
        this.first = first;
    }
}

解決后的代碼

class First {
    private Second second;

    public synchronized String getFirst() {
        second.setFirst(this);
        synchronized (this) {
            System.out.println(Thread.currentThread());
        }
        return "first";
    }
    
    public synchronized void setSecond(Second second){
        this.second = second;
    }
}

class Second {
    private First first;

    public  String getSecond() {
        first.setSecond(this);
        synchronized (this) {
            System.out.println(Thread.currentThread());
        }
        return "second";
    }
    
    public synchronized void setFirst(First first){
        this.first = first;
    }
}
3,資源死鎖

當(dāng)多個(gè)線(xiàn)程相互持有彼此正在等待的鎖而又不釋放自己已持有的鎖時(shí)會(huì)發(fā)生死鎖,當(dāng)它們?cè)谙嗤馁Y源集合上等待時(shí),也會(huì)發(fā)生死鎖.

二,死鎖的避免和診斷

1,死鎖的避免
使用Lock類(lèi)定時(shí)的tryLock功能來(lái)代替內(nèi)置鎖機(jī)制,可以檢測(cè)死鎖和從死鎖中回復(fù)過(guò)來(lái).
當(dāng)使用內(nèi)置鎖時(shí),只有沒(méi)有獲得鎖,就會(huì)永遠(yuǎn)等待下去,而顯示鎖則可以指定一個(gè)超時(shí)時(shí)限,在等待超過(guò)該時(shí)間后tryLock會(huì)返回一個(gè)失敗信息.
2,通過(guò)線(xiàn)程轉(zhuǎn)儲(chǔ)信息來(lái)分析死鎖
JVM可以通過(guò)線(xiàn)程轉(zhuǎn)儲(chǔ)(Thread Dump)來(lái)幫助識(shí)別死鎖的發(fā)生.
線(xiàn)程轉(zhuǎn)儲(chǔ)包括各個(gè)運(yùn)行中的線(xiàn)程的棧追蹤信息,這類(lèi)似于發(fā)生異常時(shí)的棧追蹤信息.線(xiàn)程轉(zhuǎn)儲(chǔ)還包括加鎖信息,例如每個(gè)線(xiàn)程持有了那些鎖,在那些棧幀中獲得這些鎖,以及被阻塞的線(xiàn)程正在等待獲取哪一個(gè)鎖.

以第一個(gè)死鎖示例(鎖順序死鎖)
輸入命令: jstack -l 16601
其中16601是進(jìn)程號(hào)

下面是部分的線(xiàn)程轉(zhuǎn)儲(chǔ)信息

........
........
Found one Java-level deadlock:
=============================
"Thread-1":
  waiting to lock monitor 0x00007fa04381beb8 (object 0x00000007aaadd8b0, a java.lang.Object),
  which is held by "Thread-0"
"Thread-0":
  waiting to lock monitor 0x00007fa04381bf68 (object 0x00000007aaadd8c0, a java.lang.Object),
  which is held by "Thread-1"

Java stack information for the threads listed above:
===================================================
"Thread-1":
    at LeftRightDeadLock.rightLeft(Test.java:54)
    - waiting to lock <0x00000007aaadd8b0> (a java.lang.Object)
    - locked <0x00000007aaadd8c0> (a java.lang.Object)
    at RightLeftThread.run(Test.java:35)
"Thread-0":
    at LeftRightDeadLock.leftRight(Test.java:46)
    - waiting to lock <0x00000007aaadd8c0> (a java.lang.Object)
    - locked <0x00000007aaadd8b0> (a java.lang.Object)
    at LeftRightThread.run(Test.java:21)

Found 1 deadlock.

參考:
<<java編發(fā)編程實(shí)戰(zhàn)>>

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末奴潘,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子掘剪,更是在濱河造成了極大的恐慌奈虾,老刑警劉巖廉赔,帶你破解...
    沈念sama閱讀 212,718評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蜡塌,死亡現(xiàn)場(chǎng)離奇詭異勿负,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)琅摩,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,683評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén)房资,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)檀头,“玉大人,你說(shuō)我怎么就攤上這事溉浙〗裕” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 158,207評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵惊奇,是天一觀的道長(zhǎng)播赁。 經(jīng)常有香客問(wèn)我,道長(zhǎng)乓序,這世上最難降的妖魔是什么坎背? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,755評(píng)論 1 284
  • 正文 為了忘掉前任得滤,我火速辦了婚禮,結(jié)果婚禮上懂更,老公的妹妹穿的比我還像新娘急膀。我一直安慰自己脖阵,他們只是感情好墅茉,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,862評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布呜呐。 她就那樣靜靜地躺著蘑辑,像睡著了一般。 火紅的嫁衣襯著肌膚如雪洋魂。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 50,050評(píng)論 1 291
  • 那天衔肢,我揣著相機(jī)與錄音角骤,去河邊找鬼心剥。 笑死,一個(gè)胖子當(dāng)著我的面吹牛优烧,可吹牛的內(nèi)容都是我干的畦娄。 我是一名探鬼主播,決...
    沈念sama閱讀 39,136評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼捍掺,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼再膳!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起不瓶,我...
    開(kāi)封第一講書(shū)人閱讀 37,882評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎熙参,沒(méi)想到半個(gè)月后麦备,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,330評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡黍匾,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,651評(píng)論 2 327
  • 正文 我和宋清朗相戀三年锐涯,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了填物。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,789評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡升薯,死狀恐怖击困,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情责语,我是刑警寧澤目派,帶...
    沈念sama閱讀 34,477評(píng)論 4 333
  • 正文 年R本政府宣布企蹭,位于F島的核電站,受9級(jí)特大地震影響谅摄,放射性物質(zhì)發(fā)生泄漏送漠。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,135評(píng)論 3 317
  • 文/蒙蒙 一代兵、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧植影,春花似錦思币、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,864評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)启涯。三九已至,卻和暖如春黎做,著一層夾襖步出監(jiān)牢的瞬間松忍,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,099評(píng)論 1 267
  • 我被黑心中介騙來(lái)泰國(guó)打工宏所, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留摊溶,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,598評(píng)論 2 362
  • 正文 我出身青樓霞玄,卻偏偏與公主長(zhǎng)得像拉岁,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子惫企,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,697評(píng)論 2 351

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

  • 1雅任、競(jìng)態(tài)條件: 定義:競(jìng)態(tài)條件指的是一種特殊的情況,在這種情況下各個(gè)執(zhí)行單元以一種沒(méi)有邏輯的順序執(zhí)行動(dòng)作沪么,從而導(dǎo)致...
    Hughman閱讀 1,286評(píng)論 0 7
  • Java8張圖 11禽车、字符串不變性 12、equals()方法殉摔、hashCode()方法的區(qū)別 13逸月、...
    Miley_MOJIE閱讀 3,697評(píng)論 0 11
  • 生命中的人流 總是 穿梭不息 就好像 風(fēng)景 裝扮著身邊的世界 總有些人 悄悄地來(lái) 默默地等 潮來(lái)潮往 人生不過(guò)生死...
    六月六的細(xì)雨閱讀 255評(píng)論 0 0
  • 已知的14位圣殿騎士團(tuán)最高大師 \ 大團(tuán)長(zhǎng)(Grand Master)碗硬。 于格·德·帕揚(yáng)(Hugues de Pa...
    闊爺閱讀 638評(píng)論 0 0
  • 我們終于老的 只能坐著,躺著 只能羨慕的看著 窗戶(hù)用玻璃弛说,把世界分割 過(guò)去是一個(gè)翰意,未來(lái)是另一個(gè) 當(dāng)我們終于老的 再...
    不忍風(fēng)塵閱讀 239評(píng)論 0 1