您好,登錄后才能下訂單哦!
volatile關鍵字經常用來修飾變量。不過,volatile本身很容易被誤用。本篇就介紹一下volatile的原理和使用方式。
在介紹volatile關鍵字原理前,我們首先要了解JVM運行時的內存分配邏輯。
對于成員變量i,它存儲在堆內存中。每個線程在運行時都會有一個自己的線程棧,線程如果要訪問類的成員變量i,會通過引用獲取到堆中變量i實際的值10,然后把這個變量值拷貝到自己的棧內存中,作為一個變量副本,之后線程便不再會與堆中的變量有實際聯系。每個線程都有一個自己的本地副本,相互隔離。線程訪問自己棧內存的效率比訪問堆的效率高。線程對變量i值的修改,只會修改自己線程副本中的值,修改結束后,在線程退出前,會把自己線程副本中的值,刷新到堆中。
保證內存可見性
對于如下代碼:
public class VolatileTest implements Runnable{ //volatile private static boolean flag = false; @Override public void run() { while (!flag){ System.out.println(Thread.currentThread().getName() +"執行中"); } System.out.println(Thread.currentThread().getName() +"執行完畢"); } //main線程 public static void main(String[] args) throws InterruptedException { new Thread(new VolatileTest(), "支線程Volatile").start(); Thread.sleep(1000); flag = true; } }
大多數時候可以正常中斷,但是一旦發送異常,便會導致線程死循環。所以需要在flag標志上加一個volatile關鍵字。對于加了volatile關鍵字的變量值,線程1修改了這個值的話,會強制將修改值直接寫入堆內存中,其他線程各自線程棧中的變量副本無效,只能去堆中取最新的變量值。多個線程之間的內存可見得以保證。
值得注意的是,volatile關鍵字不能保證原子性。
private volatile int i; i++;
i ++ 這個操作涉及到獲取值,自增和賦值3部分。無法直接完成。上面想要以volatile來實現原子性的寫法是錯誤的。
禁止指令重排
現代JVM對代碼的執行順序有一定的優化。例如:
int a = 4; int b = 5; int c = a + b;
上面3條指令進過JVM優化以后,時間的執行順序不一定是從上到下,有可能是 第二條--->第一條-->第三條。總之不會影響最終執行結果。
但是在多線程情況下,如下代碼就會有風險:
//線程1: context = loadContext(); inited = true; //線程2: while(!inited ){ } doSomething(context);
線程1的兩條語句之間沒有依賴性,經過指令重排后,有可能inited置為true以后,context還沒有初始化。線程2發現inited為true,以為初始化完成,結束循環,用時間還沒有初始化的context去執行doSomething()方法。報錯。所以我們可以用volatile關鍵字修飾inited,保證context初始化。
以上所述是小編給大家介紹的Java中的關鍵字volatile詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對億速云網站的支持!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。