Java死鎖

多線程環(huán)境下,相互或循環(huán)持有對(duì)方需要的資源颇蜡,從而導(dǎo)致死鎖价说。


相互持有并等待
循環(huán)持有并等待

解決方案:
事前:避免相互持有對(duì)方的鎖,或者循環(huán)持有下一個(gè)所需要的鎖的情況风秤。盡量將某個(gè)資源限制在線程內(nèi)部鳖目,避免資源爭(zhēng)搶。
事中:找出死鎖并強(qiáng)制其中一個(gè)退出缤弦,即可打破死鎖狀態(tài)
事后:優(yōu)化代碼领迈,如:優(yōu)化鎖順序,粗化鎖粒度等碍沐。

下面狸捅,手寫一個(gè)最簡(jiǎn)單的死鎖。

package com.zm.demo.core.concurrent;

import static java.lang.System.out;

/**
 * Created by Administrator on 2020/2/29.
 */
public class DeadLock {

    public static void main(String[]args){

        String source1= "source1";
        String source2= "source2";
        Thread t1 = new Thread(()->{
            //持有source1
            synchronized (source1){
                out.println("t1 lock " + source1);
                //申請(qǐng)source2
                synchronized (source2){
                    out.println("t1 lock " + source2);
                }
            }
        });

        Thread t2 = new Thread(()->{
            //持有source2
            synchronized (source2){
                out.println(" t2 lock " + source2);
                //申請(qǐng)source1
                synchronized (source1){
                    out.println("t2 lock " + source1);
                }
            }
        });

        t1.start();
        t2.start();


    }
}

使用jstack命令查看進(jìn)程棧信息累提,可以找出死鎖狀態(tài)的線程尘喝。

#找出當(dāng)前環(huán)境下進(jìn)程信息
D:\git\zm\demo>jps 
10704 AppMain
6864 Launcher
9624
5100 Jps
9212 RemoteMavenServer

#打印進(jìn)程棧信息
D:\git\zm\demo>jstack 10704
2020-02-29 19:13:21
Full thread dump Java HotSpot(TM) 64-Bit Server VM (25.91-b14 mixed mode):

"DestroyJavaVM" #13 prio=5 os_prio=0 tid=0x00000000021ff000 nid=0x418 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"Thread-1" #12 prio=5 os_prio=0 tid=0x00000000098dd000 nid=0x628 waiting for monitor entry [0x000000000b23f000]
   java.lang.Thread.State: BLOCKED (on object monitor)
        at com.zm.demo.core.concurrent.DeadLock.lambda$main$1(DeadLock.java:27)
        - waiting to lock <0x00000000d5cf5838> (a java.lang.String)
        - locked <0x00000000d5cf5870> (a java.lang.String)
        at com.zm.demo.core.concurrent.DeadLock$$Lambda$2/1023892928.run(Unknown Source)
        at java.lang.Thread.run(Thread.java:745)

"Thread-0" #11 prio=5 os_prio=0 tid=0x00000000098dc000 nid=0x1528 waiting for monitor entry [0x0000000009a4f000]
   java.lang.Thread.State: BLOCKED (on object monitor)
        at com.zm.demo.core.concurrent.DeadLock.lambda$main$0(DeadLock.java:18)
        - waiting to lock <0x00000000d5cf5870> (a java.lang.String)
        - locked <0x00000000d5cf5838> (a java.lang.String)
        at com.zm.demo.core.concurrent.DeadLock$$Lambda$1/1831932724.run(Unknown Source)
        at java.lang.Thread.run(Thread.java:745)

"Monitor Ctrl-Break" #10 daemon prio=5 os_prio=0 tid=0x000000000965d000 nid=0x280c runnable [0x000000000a6cf000]
   java.lang.Thread.State: RUNNABLE
        at java.net.DualStackPlainSocketImpl.accept0(Native Method)
        at java.net.DualStackPlainSocketImpl.socketAccept(DualStackPlainSocketImpl.java:131)
        at java.net.AbstractPlainSocketImpl.accept(AbstractPlainSocketImpl.java:409)
        at java.net.PlainSocketImpl.accept(PlainSocketImpl.java:199)
        - locked <0x00000000d5df5730> (a java.net.SocksSocketImpl)
        at java.net.ServerSocket.implAccept(ServerSocket.java:545)
        at java.net.ServerSocket.accept(ServerSocket.java:513)
        at com.intellij.rt.execution.application.AppMain$1.run(AppMain.java:90)
        at java.lang.Thread.run(Thread.java:745)

"Service Thread" #9 daemon prio=9 os_prio=0 tid=0x00000000095c6000 nid=0x2440 runnable [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C1 CompilerThread2" #8 daemon prio=9 os_prio=2 tid=0x0000000009551800 nid=0x14b4 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C2 CompilerThread1" #7 daemon prio=9 os_prio=2 tid=0x0000000009550800 nid=0x2a38 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C2 CompilerThread0" #6 daemon prio=9 os_prio=2 tid=0x00000000081be800 nid=0x27c0 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"Attach Listener" #5 daemon prio=5 os_prio=2 tid=0x000000000953b000 nid=0x1ba4 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"Signal Dispatcher" #4 daemon prio=9 os_prio=2 tid=0x000000000953a800 nid=0x15cc runnable [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"Finalizer" #3 daemon prio=8 os_prio=1 tid=0x0000000008199800 nid=0x2460 in Object.wait() [0x000000000952e000]
   java.lang.Thread.State: WAITING (on object monitor)
        at java.lang.Object.wait(Native Method)
        - waiting on <0x00000000d5c08ee0> (a java.lang.ref.ReferenceQueue$Lock)
        at java.lang.ref.ReferenceQueue.remove(ReferenceQueue.java:143)
        - locked <0x00000000d5c08ee0> (a java.lang.ref.ReferenceQueue$Lock)
        at java.lang.ref.ReferenceQueue.remove(ReferenceQueue.java:164)
        at java.lang.ref.Finalizer$FinalizerThread.run(Finalizer.java:209)

"Reference Handler" #2 daemon prio=10 os_prio=2 tid=0x0000000008152800 nid=0x233c in Object.wait() [0x000000000932f000]
   java.lang.Thread.State: WAITING (on object monitor)
        at java.lang.Object.wait(Native Method)
        - waiting on <0x00000000d5c06b50> (a java.lang.ref.Reference$Lock)
        at java.lang.Object.wait(Object.java:502)
        at java.lang.ref.Reference.tryHandlePending(Reference.java:191)
        - locked <0x00000000d5c06b50> (a java.lang.ref.Reference$Lock)
        at java.lang.ref.Reference$ReferenceHandler.run(Reference.java:153)

"VM Thread" os_prio=2 tid=0x000000000814b000 nid=0x2b4c runnable

"GC task thread#0 (ParallelGC)" os_prio=0 tid=0x000000000224d000 nid=0x2a34 runnable

"GC task thread#1 (ParallelGC)" os_prio=0 tid=0x000000000224e800 nid=0x21f4 runnable

"GC task thread#2 (ParallelGC)" os_prio=0 tid=0x0000000002250000 nid=0x1604 runnable

"GC task thread#3 (ParallelGC)" os_prio=0 tid=0x0000000002251800 nid=0x1aec runnable

"VM Periodic Task Thread" os_prio=2 tid=0x000000000960a800 nid=0x1a78 waiting on condition

JNI global references: 317


Found one Java-level deadlock:
=============================
"Thread-1":
  waiting to lock monitor 0x0000000008156898 (object 0x00000000d5cf5838, a java.lang.String),
  which is held by "Thread-0"
"Thread-0":
  waiting to lock monitor 0x0000000008159338 (object 0x00000000d5cf5870, a java.lang.String),
  which is held by "Thread-1"

Java stack information for the threads listed above:
===================================================
"Thread-1":
        at com.zm.demo.core.concurrent.DeadLock.lambda$main$1(DeadLock.java:27)
        - waiting to lock <0x00000000d5cf5838> (a java.lang.String)
        - locked <0x00000000d5cf5870> (a java.lang.String)
        at com.zm.demo.core.concurrent.DeadLock$$Lambda$2/1023892928.run(Unknown Source)
        at java.lang.Thread.run(Thread.java:745)
"Thread-0":
        at com.zm.demo.core.concurrent.DeadLock.lambda$main$0(DeadLock.java:18)
        - waiting to lock <0x00000000d5cf5870> (a java.lang.String)
        - locked <0x00000000d5cf5838> (a java.lang.String)
        at com.zm.demo.core.concurrent.DeadLock$$Lambda$1/1831932724.run(Unknown Source)
        at java.lang.Thread.run(Thread.java:745)

Found 1 deadlock.

最后幾行,很明顯的斋陪,直接打印出死鎖信息了朽褪。而且直接打印出死鎖的代碼位置,我們可以很方便的找到代碼并進(jìn)行優(yōu)化无虚。
Thread-0持有0x00000000d5cf5838缔赠,等待0x00000000d5cf5870
同時(shí)
Thread-1持有0x00000000d5cf5870,等待0x00000000d5cf5838
因此友题,產(chǎn)生死鎖嗤堰。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市度宦,隨后出現(xiàn)的幾起案子踢匣,更是在濱河造成了極大的恐慌,老刑警劉巖戈抄,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件符糊,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡呛凶,警方通過查閱死者的電腦和手機(jī)男娄,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人模闲,你說我怎么就攤上這事建瘫。” “怎么了尸折?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵啰脚,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我实夹,道長(zhǎng)橄浓,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任亮航,我火速辦了婚禮荸实,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘缴淋。我一直安慰自己准给,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布重抖。 她就那樣靜靜地躺著露氮,像睡著了一般。 火紅的嫁衣襯著肌膚如雪钟沛。 梳的紋絲不亂的頭發(fā)上畔规,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天,我揣著相機(jī)與錄音恨统,去河邊找鬼油讯。 笑死,一個(gè)胖子當(dāng)著我的面吹牛延欠,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播沈跨,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼由捎,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了饿凛?” 一聲冷哼從身側(cè)響起狞玛,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎涧窒,沒想到半個(gè)月后心肪,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡纠吴,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年硬鞍,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡固该,死狀恐怖锅减,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情伐坏,我是刑警寧澤怔匣,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布,位于F島的核電站桦沉,受9級(jí)特大地震影響每瞒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜纯露,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一剿骨、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧苔埋,春花似錦懦砂、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至玉工,卻和暖如春羽资,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背遵班。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工屠升, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人狭郑。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓腹暖,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親翰萨。 傳聞我的和親對(duì)象是個(gè)殘疾皇子脏答,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354