史上最通俗理解的Java死鎖代碼演示
死鎖的概念
知識儲備
對象鎖:Java一切皆對象,每個類都有一個class文件。由class文件可以new出對象,我們簡單認(rèn)識 下java對象,對象有個對象頭信息,也就是這個對象概述,其中一條信息就是對象鎖,也就是我們當(dāng)前對象有沒有被鎖定,被哪個引用鎖定。
synchronized:synchronized是java關(guān)鍵詞,如果運用到方法上代表我們鎖的是這個方法,如果我們鎖的代碼塊,代表再這個代碼塊內(nèi)我們持有這個鎖,Java Effective也是提倡減小鎖的范圍。我們進入同步代碼塊會加鎖,執(zhí)行完同步代碼塊會釋放鎖。
死鎖:通俗理解為死掉的鎖。如果沒有死掉的鎖它的聲明周期是:持有鎖->釋放鎖。死后我們可以理解為持有鎖但是不釋放鎖,也就是我們同步代碼塊沒有執(zhí)行完?我們只需要分析同步代碼塊的哪里沒有執(zhí)行就好了,看下面一個例子
演示死鎖
package com.yang.kuangTeacher; import java.util.concurrent.TimeUnit; /** * @author: fudy * @date: 2020/9/13 下午 12:21 * @Decription: 演示死鎖(內(nèi)容參考B站狂神說JAVA) **/ public class DeadLock { public static void main(String[] args) { MarkUp markUp0 = new MarkUp("迪麗熱巴",0); MarkUp markUp1 = new MarkUp("楊冪",1); markUp0.start(); markUp1.start(); } } // 口紅類 class LipStick { } // 鏡子類 class Mirror { } // 化妝類 class MarkUp extends Thread { private int choice; private String userName; private static LipStick lipStick = new LipStick(); private static Mirror mirror = new Mirror(); MarkUp(String userName, int choice) { this.userName = userName; this.choice = choice; } @Override public void run() { try { markUP(); } catch (InterruptedException e) { e.printStackTrace(); } } private void markUP() throws InterruptedException { // 如果選擇0方式化妝 if (choice == 0) { // 同步代碼塊的鎖,在同步代碼塊有效 synchronized (lipStick) { System.out.println(userName + "拿到了口紅"); // 拿到口紅后再拿鏡子 TimeUnit.SECONDS.sleep(1); // 程序執(zhí)行此處會停止 -----------------這里死鎖-----------------------> synchronized (mirror) { System.out.println(userName + "拿到了鏡子"); } } } // 如果選擇1方式化妝 if (choice == 1) { // 同步代碼塊的鎖,在同步代碼塊有效 synchronized (mirror) { System.out.println(userName + "拿到了鏡子"); // 拿到鏡子后再拿口紅 TimeUnit.SECONDS.sleep(1); // 程序執(zhí)行此處會停止 -----------------這里死鎖-----------------------> synchronized (lipStick) { System.out.println(userName + "拿到了口紅"); } } } } }
我們剛才認(rèn)為,死鎖是由于同步代碼塊沒有執(zhí)行完,導(dǎo)致不會釋放鎖,我們分析以上兩個死鎖的原因。
- 在線程1方式0化妝中由于我們拿到了口紅鎖后,睡眠一秒鐘(有可能先執(zhí)行線程2)
- 在線程2方式1化妝中由于我們拿到了鏡子鎖后,睡眠一秒鐘
假如線程1先獲得口紅鎖執(zhí)行完畢,準(zhǔn)備拿鏡子鎖時,發(fā)現(xiàn)鏡子對象被持有了,所以他會等待鏡子鎖被釋放。
線程2先執(zhí)行獲得鏡子鎖完畢,準(zhǔn)備拿口紅鎖時,發(fā)現(xiàn)口紅對象被持有了,所以他會等待口紅鎖被釋放。
如果我們不關(guān)閉程序,兩個線程會一直等待下去。我們可以理解為死鎖,無法釋放鎖。
解決死鎖
在上述例子中,我們因為想同時拿到兩個鎖去做一件事情才會導(dǎo)致死鎖,按照J(rèn)ava Effective提倡減小鎖的范圍,我們對問題進行改進。
我們可以拿到口紅鎖后執(zhí)行口紅方法后釋放口紅鎖,想要鏡子鎖再同步代碼塊拿鏡子鎖即可。
package com.yang.kuangTeacher; import java.util.concurrent.TimeUnit; /** * @author: fudy * @date: 2020/9/13 下午 12:21 * @Decription: 演示死鎖(內(nèi)容參考B站狂神說JAVA) **/ public class DeadLock { public static void main(String[] args) { MarkUp markUp0 = new MarkUp("迪麗熱巴",0); MarkUp markUp1 = new MarkUp("楊冪",1); markUp0.start(); markUp1.start(); } } // 口紅類 class LipStick { } // 鏡子類 class Mirror { } // 化妝類 class MarkUp extends Thread { private int choice; private String userName; private static LipStick lipStick = new LipStick(); private static Mirror mirror = new Mirror(); MarkUp(String userName, int choice) { this.userName = userName; this.choice = choice; } @Override public void run() { try { markUP(); } catch (InterruptedException e) { e.printStackTrace(); } } private void markUP() throws InterruptedException { // 如果選擇0方式化妝 if (choice == 0) { // 同步代碼塊的鎖,在同步代碼塊有效 synchronized (lipStick) { System.out.println(userName + "拿到了口紅"); TimeUnit.SECONDS.sleep(1); } // 拿到口紅后再拿鏡子 ------------------------改進--------------------------- synchronized (mirror) { System.out.println(userName + "拿到了鏡子"); } } // 如果選擇1方式化妝 if (choice == 1) { // 同步代碼塊的鎖,在同步代碼塊有效 synchronized (mirror) { System.out.println(userName + "拿到了鏡子"); TimeUnit.SECONDS.sleep(1); } // 拿到鏡子后再拿口紅 ------------------------改進--------------------------- synchronized (lipStick) { System.out.println(userName + "拿到了口紅"); } } } }
通過及時釋放鎖,也就是縮小同步代碼塊的范圍,我們使用鎖結(jié)束后及時釋放,這樣是一種解決死鎖的方式,通過這個例子我們以后編寫代碼就會警惕鎖的同步代碼的范圍。
總結(jié)
到此這篇關(guān)于Java死鎖代碼演示的文章就介紹到這了,更多相關(guān)Java死鎖代碼演示內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Springboot+mybatis-plus+注解實現(xiàn)數(shù)據(jù)權(quán)限隔離
本文將結(jié)合實例代碼,介紹Springboot+mybatis-plus+注解實現(xiàn)數(shù)據(jù)權(quán)限隔離,文中通過示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-07-07spring?boot整合mongo查詢converter異常排查記錄
這篇文章主要為大家介紹了spring?boot整合mongo查詢時拋出converter異常的排查解決記錄,有需要的朋友可以借鑒參考下,希望能夠有所幫助2022-03-03詳細(xì)談?wù)凧ava中l(wèi)ong和double的原子性
原子性是指一個操作或多個操作要么全部執(zhí)行,且執(zhí)行的過程不會被任何因素打斷,要么就都不執(zhí)行,下面這篇文章主要給大家介紹了關(guān)于Java中l(wèi)ong和double原子性的相關(guān)資料,需要的朋友可以參考下2021-08-08詳解java如何實現(xiàn)將數(shù)據(jù)導(dǎo)出為yaml
這篇文章主要為大家詳細(xì)介紹了java如何利用snakeyaml和freemarker實現(xiàn)將數(shù)據(jù)導(dǎo)出為yaml文件,文中的示例代碼講解詳細(xì),有需要的小伙伴可以參考一下2023-11-11