欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

史上最通俗理解的Java死鎖代碼演示

 更新時間:2020年09月15日 08:59:21   作者:三少說  
這篇文章主要給大家介紹了關(guān)于Java死鎖代碼演示的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

死鎖的概念

知識儲備

對象鎖: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)限隔離

    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-07
  • Java操作集合工具類Collections使用詳解

    Java操作集合工具類Collections使用詳解

    這篇文章主要介紹了java操作集合工具類Collections使用詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-12-12
  • SpringCloud Bus 消息總線的具體使用

    SpringCloud Bus 消息總線的具體使用

    這篇文章主要介紹了SpringCloud Bus 消息總線的具體使用,詳細(xì)的介紹了什么是消息總線以及具體配置,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-12-12
  • spring?boot整合mongo查詢converter異常排查記錄

    spring?boot整合mongo查詢converter異常排查記錄

    這篇文章主要為大家介紹了spring?boot整合mongo查詢時拋出converter異常的排查解決記錄,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2022-03-03
  • 詳細(xì)談?wù)凧ava中l(wèi)ong和double的原子性

    詳細(xì)談?wù)凧ava中l(wèi)ong和double的原子性

    原子性是指一個操作或多個操作要么全部執(zhí)行,且執(zhí)行的過程不會被任何因素打斷,要么就都不執(zhí)行,下面這篇文章主要給大家介紹了關(guān)于Java中l(wèi)ong和double原子性的相關(guān)資料,需要的朋友可以參考下
    2021-08-08
  • 詳解Struts2攔截器機制

    詳解Struts2攔截器機制

    這篇文章主要介紹了詳解Struts2攔截器機制,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-06-06
  • Java日常練習(xí)題,每天進步一點點(41)

    Java日常練習(xí)題,每天進步一點點(41)

    下面小編就為大家?guī)硪黄狫ava基礎(chǔ)的幾道練習(xí)題(分享)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧,希望可以幫到你
    2021-07-07
  • 詳解java如何實現(xiàn)將數(shù)據(jù)導(dǎo)出為yaml

    詳解java如何實現(xiàn)將數(shù)據(jù)導(dǎo)出為yaml

    這篇文章主要為大家詳細(xì)介紹了java如何利用snakeyaml和freemarker實現(xiàn)將數(shù)據(jù)導(dǎo)出為yaml文件,文中的示例代碼講解詳細(xì),有需要的小伙伴可以參考一下
    2023-11-11
  • Springboot的啟動原理詳細(xì)解讀

    Springboot的啟動原理詳細(xì)解讀

    這篇文章主要介紹了Springboot的啟動原理詳細(xì)解讀,springboot項目一般都是打包成jar包直接運行main方法啟動,當(dāng)然也可以跟傳統(tǒng)的項目一樣打包war包放在tomcat里面啟動.那么springboot怎么直接通過main方法啟動呢,需要的朋友可以參考下
    2023-11-11
  • Java如何基于IO流實現(xiàn)同一文件讀寫操作

    Java如何基于IO流實現(xiàn)同一文件讀寫操作

    這篇文章主要介紹了Java如何基于IO流實現(xiàn)文件讀寫操作,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-10-10

最新評論