史上最通俗理解的Java死鎖代碼演示
死鎖的概念
知識儲備
對象鎖:Java一切皆對象,每個類都有一個class文件。由class文件可以new出對象,我們簡單認識 下java對象,對象有個對象頭信息,也就是這個對象概述,其中一條信息就是對象鎖,也就是我們當前對象有沒有被鎖定,被哪個引用鎖定。
synchronized:synchronized是java關鍵詞,如果運用到方法上代表我們鎖的是這個方法,如果我們鎖的代碼塊,代表再這個代碼塊內我們持有這個鎖,Java Effective也是提倡減小鎖的范圍。我們進入同步代碼塊會加鎖,執行完同步代碼塊會釋放鎖。
死鎖:通俗理解為死掉的鎖。如果沒有死掉的鎖它的聲明周期是:持有鎖->釋放鎖。死后我們可以理解為持有鎖但是不釋放鎖,也就是我們同步代碼塊沒有執行完?我們只需要分析同步代碼塊的哪里沒有執行就好了,看下面一個例子
演示死鎖
package com.yang.kuangTeacher;import java.util.concurrent.TimeUnit;/** * @author: fudy * @date: 2020/9/13 下午 12:21 * @Decription: 演示死鎖(內容參考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 + '拿到了口紅'); } } } }}
我們剛才認為,死鎖是由于同步代碼塊沒有執行完,導致不會釋放鎖,我們分析以上兩個死鎖的原因。
在線程1方式0化妝中由于我們拿到了口紅鎖后,睡眠一秒鐘(有可能先執行線程2) 在線程2方式1化妝中由于我們拿到了鏡子鎖后,睡眠一秒鐘假如線程1先獲得口紅鎖執行完畢,準備拿鏡子鎖時,發現鏡子對象被持有了,所以他會等待鏡子鎖被釋放。
線程2先執行獲得鏡子鎖完畢,準備拿口紅鎖時,發現口紅對象被持有了,所以他會等待口紅鎖被釋放。
如果我們不關閉程序,兩個線程會一直等待下去。我們可以理解為死鎖,無法釋放鎖。
解決死鎖
在上述例子中,我們因為想同時拿到兩個鎖去做一件事情才會導致死鎖,按照Java Effective提倡減小鎖的范圍,我們對問題進行改進。
我們可以拿到口紅鎖后執行口紅方法后釋放口紅鎖,想要鏡子鎖再同步代碼塊拿鏡子鎖即可。
package com.yang.kuangTeacher;import java.util.concurrent.TimeUnit;/** * @author: fudy * @date: 2020/9/13 下午 12:21 * @Decription: 演示死鎖(內容參考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 + '拿到了口紅'); } } }}
通過及時釋放鎖,也就是縮小同步代碼塊的范圍,我們使用鎖結束后及時釋放,這樣是一種解決死鎖的方式,通過這個例子我們以后編寫代碼就會警惕鎖的同步代碼的范圍。
總結
到此這篇關于Java死鎖代碼演示的文章就介紹到這了,更多相關Java死鎖代碼演示內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!
相關文章: