您好,登錄后才能下訂單哦!
本篇文章給大家分享的是有關Java中如何應用volatile關鍵字,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。
Jav內存模型(JMM)
談到多線程就應該了解一下Java內存模型(JMM)的抽象示意圖.下圖:
線程A和線程B執行的是時候,會去讀取共享變量(臨界區),然后各自拷貝一份回到自己的本地內存,執行后續操作。JMM模型是一種規范,就像Java的接口一樣。JMM會涉及到三個問題:原子性,可見性,有序性。所謂原子性。就是說一個線程的執行會不會被其他線程影響的。他是不可中斷的。舉個例子:
int i=1
這個語句在Jmm中就是原子性的。無論是一個線程執行還是多個線程執行這個語句,讀出來的i就是等于1。那什么是非原子性呢,按道理如果Java的代碼都是原子性,應該就不會有線程問題了啊。其實JMM這是規定某些語句是原子性罷了。舉個非原子性例子:
i ++;
這個操作就不是原子性的了。因為他就是包含了三個操作:第一讀取i的值,第二將i加上1,第三將結果賦值回來給i,更新i的值。所謂可見性。可見性表示如果一個值在線程A修改了,線程B就會馬上知道這個結果。
所謂有序性。所謂有序性值的是語意的有序性。就是說代碼順序可能會發生變化。因為有一個指令重排機制。所謂指令重排,他會改變代碼執行順序,為了讓cpu執行效率更高。為了防止重排序出錯,JMM有個happen-before規則,這個規則限制了那些語句執行在前,那些語句執行在后。
Happen-before:
程序順序原則:一個線程內保證語義的串行性
volatile原則:volatile變量的寫發生在讀之前
鎖規則:先加鎖再解鎖
傳遞性:a先于b,b先于c,則a必定先于c
線程的start方法先于他的每一個操作
線程所有的操作先于線程的終結
對象的構造函數執行、結束先于finalize()方法。
volatile
進入正題,volatile可以保證變量(臨界區)的可見性以及有序性,但是不能保證原子性。舉個例子:
public class VolatileTest implements Runnable{ private static VolatileTest volatileTest = new VolatileTest(); private static volatile int i= 0; public static void main(String[] args) throws InterruptedException { for (int j = 0; j < 20; j++) { Thread a = new Thread(new VolatileTest()); Thread b = new Thread(new VolatileTest()); a.start();b.start(); a.join();b.join(); System.out.print(i+"&&"); } } @Override public void run() { for (int j = 0; j < 1000; j++) { i++; } }}// 輸出結果// 2000&&4000&&5852&&7852&&9852&&11852&&13655&&15655&&17655&&19655&&21306 //&&22566&&24566&&26189&&28189&&30189&&32189&&34189&&36189&&38089&&
有結果看到有問題,雖然i已經添加了volatile關鍵字,說明volatile關鍵字不能保證i++的原子性。
那什么場景適合使用volatile關鍵字
1、輕量級的“讀-寫鎖”策略
private volatile int value;public int getValue(){ return value;}public synchronized void doubleValue(){ value = value*value; }
2.單例模式(雙檢查鎖機制
private volatile static Singleton instace; public static Singleton getInstance(){ // 沒有使用同步方法,而是同步方法塊 //第一次null檢查 ,利用volatile的線程間可見性,不需要加鎖,性能提高 if(instance == null){ synchronized(Singleton.class) { //鎖住類對象,阻塞其他線程 //第二次null檢查,以保證不會創建重復的實例 if(instance == null){ instance = new Singleton(); // 禁止重排序 } } } return instance;
以上就是Java中如何應用volatile關鍵字,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。