亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java并發synchronized用在方法上鎖住的內容是什么

發布時間:2021-09-04 14:17:06 來源:億速云 閱讀:127 作者:chen 欄目:編程語言

本篇內容主要講解“Java并發synchronized用在方法上鎖住的內容是什么”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“Java并發synchronized用在方法上鎖住的內容是什么”吧!

synchronized用在方法上鎖住的是什么?

鎖住的是當前對象的當前方法,會使得其他線程訪問該對象的synchronized方法或者代碼塊阻塞,但并不會阻塞非synchronized方法。

臟讀

一個常見的概念。在多線程中,難免會出現在多個線程中對同一個對象的實例變量或者全局靜態變量進行并發訪問的情況,如果不做正確的同步處理,那么產生的后果就是"臟讀",也就是取到的數據其實是被更改過的。注意這里 局部變量是不存在臟讀的情況

public class ThreadDomain13
{
  private int num = 0;
  
  public void addNum(String userName)
  {
    try
    {
      if ("a".equals(userName))
      {
        num = 100;
        System.out.println("a set over!");
        Thread.sleep(2000);
      }
      else
      {
        num = 200;
        System.out.println("b set over!");
      }
      System.out.println(userName + " num = " + num);
    }
    catch (InterruptedException e)
    {
      e.printStackTrace();
    }
  }
}

寫兩個線程分別去add字符串"a"和字符串"b":

public class MyThread13_0 extends Thread
{
  private ThreadDomain13 td;
  
  public MyThread13_0(ThreadDomain13 td)
  {
    this.td = td;
  }
  
  public void run()
  {
    td.addNum("a");
  }
}
public class MyThread13_1 extends Thread
{
  private ThreadDomain13 td;
  
  public MyThread13_1(ThreadDomain13 td)
  {
    this.td = td;
  }
  
  public void run()
  {
    td.addNum("b");
  }
}

寫一個主函數分別運行這兩個線程:

public static void main(String[] args)
{
  ThreadDomain13 td = new ThreadDomain13();
  MyThread13_0 mt0 = new MyThread13_0(td);
  MyThread13_1 mt1 = new MyThread13_1(td);
  mt0.start();
  mt1.start();
}
//看一下運行結果
a set over!
b set over!
b num = 200
a num = 200

按照正常來看應該打印"a num = 100"和"b num = 200"才對,現在卻打印了"b num = 200"和"a num = 200",這就是線程安全問題。我們可以想一下是怎么會有線程安全的問題的:

1、mt0先運行,給num賦值100,然后打印出"a set over!",開始睡覺

2、mt0在睡覺的時候,mt1運行了,給num賦值200,然后打印出"b set over!",然后打印"b num = 200"

3、mt1睡完覺了,由于mt0的num和mt1的num是同一個num,所以mt1把num改為了200了,mt0也沒辦法,對于它來說,num只能是100,mt0繼續運行代碼,打印出"a num = 200"

分析了產生問題的原因,解決就很簡單了,給addNum(String userName)方法加同步即可:

多線程線synchronized關鍵字加到方法上

public class ThreadDomain13
{
  private int num = 0;
  
  public synchronized void addNum(String userName)
  {
    try
    {
      if ("a".equals(userName))
      {
        num = 100;
        System.out.println("a set over!");
        Thread.sleep(2000);
      }
      else
      {
        num = 200;
        System.out.println("b set over!");
      }
      System.out.println(userName + " num = " + num);
    }
    catch (InterruptedException e)
    {
      e.printStackTrace();
    }
  }
}

看一下運行結果:

a set over!
a num = 100
b set over!
b num = 200

多個對象多個鎖

在同步的情況下,把main函數內的代碼改一下:

public static void main(String[] args)
{
  ThreadDomain13 td0 = new ThreadDomain13();
  ThreadDomain13 td1 = new ThreadDomain13();
  MyThread13_0 mt0 = new MyThread13_0(td0);
  MyThread13_1 mt1 = new MyThread13_1(td1);
  mt0.start();
  mt1.start();
}

看一下運行結果:

a set over!
b set over!
b num = 200
a num = 100

這里有一個重要的概念。關鍵字synchronized取得的鎖都是對象鎖,而不是把一段代碼或方法(函數)當作鎖,這里如果是把一段代碼或方法(函數)當作鎖,其實獲取的也是對象鎖,只是監視器(對象)不同而已,哪個線程先執行帶synchronized關鍵字的方法,哪個線程就持有該方法所屬對象的鎖,其他線程都只能呈等待狀態。但是這有個前提:既然鎖叫做對象鎖,那么勢必和對象相關,所以多個線程訪問的必須是同一個對象。

如果多個線程訪問的是多個對象,那么Java虛擬機就會創建多個鎖,就像上面的例子一樣,創建了兩個ThreadDomain13對象,就產生了2個鎖。既然兩個線程持有的是不同的鎖,自然不會受到"等待釋放鎖"這一行為的制約,可以分別運行addNum(String userName)中的代碼。

synchronized(this)鎖住的是什么?

鎖住的是當前的對象。當synchronized塊里的內容執行完之后,釋放當前對象的鎖。同一時刻若有多個線程訪問這個對象,則會被阻塞。

synchronized(object)鎖住的什么?

鎖住的是object對象。當synchronized塊里的內容執行完之后,釋放object對象的鎖。同一時刻若有多個線程訪問這個對象,則會被阻塞。

這里需要注意的是如果object為Integer、String等等包裝類時(new出的對象除外),并不會鎖住當前對象,也不會阻塞線程。因為包裝類是final的,不可修改的,如果修改則會生成一個新的對象。所以,在一個線程對其進行修改后,其他線程在獲取該對象的鎖時,該對象已經不是原來的那個對象,所以獲取到的是另一個對象的鎖,所以不會產生阻塞。

到此,相信大家對“Java并發synchronized用在方法上鎖住的內容是什么”有了更深的了解,不妨來實際操作一番吧!這里是億速云網站,更多相關內容可以進入相關頻道進行查詢,關注我們,繼續學習!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

岳普湖县| 义马市| 四平市| 阿克陶县| 中阳县| 民勤县| 乐东| 瓦房店市| 融水| 西城区| 阿图什市| 博客| 奎屯市| 西昌市| 松溪县| 鲜城| 巢湖市| 永宁县| 宿松县| 托克托县| 合江县| 石泉县| 尤溪县| 东乡族自治县| 永胜县| 长沙县| 江达县| 镇宁| 玛纳斯县| 安吉县| 枝江市| 阳西县| 南皮县| 右玉县| 阜康市| 蓬安县| 湘潭市| 清河县| 邵东县| 吉林省| 城口县|