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

Java volatile關鍵字特性講解上篇

 更新時間:2022年12月12日 15:52:09   作者:愛吃南瓜糕的北絡  
JMM要求保證可見性、原子性、有序性,volatile可以保證其中的兩個,本篇文章具體驗證volatile的可見性,不原子性和禁重排,同時解決volatile的不保證原子性,讓代碼具有原子性

一、概述

volatile是Java中的關鍵字,用來修飾會被不同線程訪問和修改的變量。

volatile是Java虛擬機提供的輕量級的同步機制,它有三個特性:

(1)保證可見性

(2)不保證原子性

(3)禁止指令重排

二、特性詳解

volatile保證可見性

Java內存模型(JMM)定義了一組規(guī)則、規(guī)范,規(guī)定了程序中各個變量的訪問方法。JMM關于同步的規(guī)定:

(1)線程解鎖前,必須把共享變量的值刷新回主內存;

(2)線程加鎖前,必須讀取主內存的最新值同步到自己的工作內存;

(3)加鎖解鎖必須是同一把鎖;

說明:由于JVM運行程序的實體是線程,創(chuàng)建每個線程時,JMM會為其創(chuàng)建一個工作內存(也稱棧空間),工作內存是每個線程的私有數據區(qū)域。

Java內存模型規(guī)定所有變量都存儲在主內存,主內存是共享內存區(qū)域,所有線程都可以訪問。

但是線程對變量的操作(讀取、賦值等)必須在工作內存中進行。因此首先要將變量從主內存拷貝到自己的工作內存,然后對變量進行操作,操作完成后再將變量寫會主內存中。

舉例說明:

(1)火車票賣票系統(tǒng)還剩下一張票,并已經刷入到主內存中:ticketNum = 1;

(2)此時有3個用戶在同時購買票,3個線程都讀入了目前的票數,ticketNum=1,那么線程就會繼續(xù)進入購買流程。

(3)假設其中一個線程先搶占了CPU資源,先買到票,并將自己的工作內存中的ticketNum值改為0,ticketNum=0,然后再寫回到主內存。

這時,由于一個線程的用戶已經買到了票,那么其他用戶的線程應該不能再繼續(xù)進入購買票的流程了,因此需要系統(tǒng)通知到其他線程 ticketNum=0 這個消息。如果可以達到這樣的效果,可以理解為 具有可見性。

無可見性代碼演示:

@Test
public void test1() {
    DataDemo dataDemo = new DataDemo();
    RunThread runThread = new RunThread(dataDemo);
    runThread.start();
    while (dataDemo.getNumber() == 0) {
    }
    System.out.println("具有可見性驗證通過");
}
public class DataDemo {
    private int number = 0;
    public void add() {
        this.number = this.number + 10;
    }
    public int getNumber() {
        return number;
    }
}
public class RunThread extends Thread {
    private DataDemo dataDemo;
    public RunThread(DataDemo dataDemo) {
        this.dataDemo = dataDemo;
    }
    @Override
    public void run() {
        System.out.println("線程[" + Thread.currentThread().getName() + "] 正在執(zhí)行");
        try {
            TimeUnit.SECONDS.sleep(3);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        dataDemo.add();
        System.out.println("線程[" + Thread.currentThread().getName() + "]更新后,number值為:" + dataDemo.getNumber());
    }
}

執(zhí)行結果:
線程[Thread-0] 正在執(zhí)行
線程[Thread-0]更新后,number值為:10

結果分析:

可以看出子線程啟動后將number值改為了10,雖然已經改為了非0,但是主線程仍然一直處于while循環(huán)中,因此此時number不具有可見性,系統(tǒng)不會主動通知主線程number值修改。

原理說明:

這個問題其實就是私有堆棧中的值和公共堆棧中的值不同步造成的。解決這樣的問題就要使用 volatile 關鍵字了,它主要的作用就是當線程訪問number這個變量時,強制性從公共堆棧中進行取值。

可見性代碼演示:

@Test
public void test1() {
    DataDemo dataDemo = new DataDemo();
    RunThread runThread = new RunThread(dataDemo);
    runThread.start();
    while (dataDemo.getNumber() == 0) {
    }
    System.out.println("具有可見性驗證通過");
}
public class DataDemo {
    // 給變量 number 添加 volatile 關鍵字修飾
    volatile private int number = 0;
    public void add() {
        this.number = this.number + 10;
    }
    public int getNumber() {
        return number;
    }
}
public class RunThread extends Thread {
    private DataDemo dataDemo;
    public RunThread(DataDemo dataDemo) {
        this.dataDemo = dataDemo;
    }
    @Override
    public void run() {
        System.out.println("線程[" + Thread.currentThread().getName() + "] 正在執(zhí)行");
        try {
            TimeUnit.SECONDS.sleep(3);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        dataDemo.add();
        System.out.println("線程[" + Thread.currentThread().getName() + "]更新后,number值為:" + dataDemo.getNumber());
    }
}

執(zhí)行結果:
線程[Thread-0] 正在執(zhí)行
線程[Thread-0]更新后,number值為:10
具有可見性驗證通過

結果分析:

通過對變量number變量添加了volatile關鍵字修飾,可以看出子線程啟動后將number值改為了10,隨后主線程跳出了while循環(huán),輸出了“具有可見性驗證通過”,說明此時number具有可見性。

原理說明:

通過使用 volatile 關鍵字,強制從公共內存中讀取變量的值,內存結構如圖:

到此這篇關于Java volatile關鍵字特性講解上篇的文章就介紹到這了,更多相關Java volatile內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • Java中的轉換流InputStreamReader解讀

    Java中的轉換流InputStreamReader解讀

    InputStreamReader是Java.io包中的一個類,用于將字節(jié)輸入流轉換為字符輸入流,它繼承自java.io.Reader類,提供了兩種構造方法,可以使用默認或指定字符集創(chuàng)建實例,常用方法包括讀取字符、判斷是否準備好讀取數據和關閉流
    2024-09-09
  • Java中Integer的parseInt和valueOf的區(qū)別詳解

    Java中Integer的parseInt和valueOf的區(qū)別詳解

    這篇文章主要介紹了Java中Integer的parseInt和valueOf的區(qū)別詳解,nteger.parseInt(s)是把字符串解析成int基本類型,Integer.valueOf(s)是把字符串解析成Integer對象類型,其實int就是Integer解包裝,Integer就是int的包裝,需要的朋友可以參考下
    2023-11-11
  • SpringBoot?Security使用MySQL實現驗證與權限管理

    SpringBoot?Security使用MySQL實現驗證與權限管理

    安全管理是軟件系統(tǒng)必不可少的的功能。根據經典的“墨菲定律”——凡是可能,總會發(fā)生。如果系統(tǒng)存在安全隱患,最終必然會出現問題,這篇文章主要介紹了SpringBoot安全管理Spring?Security基本配置
    2022-11-11
  • 超詳細講解SpringCloud?Commons公共抽象的用法

    超詳細講解SpringCloud?Commons公共抽象的用法

    這篇文章主要介紹了超詳細講解SpringCloud?Commons公共抽象的用法,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • Java不用算數運算符來實現求和方法

    Java不用算數運算符來實現求和方法

    我們都知道,Java的運算符除了具有優(yōu)先級之外,還有一個結合性的特點。當一個表達式中出現多種運算符時,執(zhí)行的先后順序不僅要遵守運算符優(yōu)先級別的規(guī)定,還要受運算符結合性的約束,以便確定是自左向右進行運算還是自右向左進行運算,但是如果不用運算符怎么求和呢
    2022-04-04
  • mybatis-plus支持null字段全量更新的兩種方法

    mybatis-plus支持null字段全量更新的兩種方法

    本文主要介紹了mybatis-plus支持null字段全量更新的兩種方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-02-02
  • 文件上傳SpringBoot后端MultipartFile參數報空問題的解決辦法

    文件上傳SpringBoot后端MultipartFile參數報空問題的解決辦法

    這篇文章主要介紹了文件上傳SpringBoot后端MultipartFile參數報空問題的解決辦法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-11-11
  • SpringBoot通過注解注入Bean的幾種方式解析

    SpringBoot通過注解注入Bean的幾種方式解析

    這篇文章主要為大家介紹了SpringBoot注入Bean的幾種方式示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步早日升職加薪
    2022-03-03
  • springboot結合mysql主從來實現讀寫分離的方法示例

    springboot結合mysql主從來實現讀寫分離的方法示例

    這篇文章主要介紹了springboot結合mysql主從來實現讀寫分離的方法示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-04-04
  • Java讀取并下載網絡文件的方法

    Java讀取并下載網絡文件的方法

    這篇文章主要為大家詳細介紹了Java讀取并下載網絡文件的方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-08-08

最新評論