首頁  >  文章  >  Java  >  Java volatile關鍵字的程式碼解析

Java volatile關鍵字的程式碼解析

不言
不言轉載
2018-10-22 14:43:322289瀏覽

這篇文章帶給大家的內容是關於Java volatile關鍵字的程式碼解析,有一定的參考價值,有需要的朋友可以參考一下,希望對你有幫助。

相信大多數Java程式設計師都學習過volatile這個關鍵字的用法。百度百科上對volatile的定義:

volatile是一個類型修飾符(type specifier),被設計用來修飾被不同執行緒存取和修改的變數。 volatile的作用是作為指令關鍵字,確保本指令不會因編譯器的最佳化而省略,且要求每次直接讀值。

可能有很多剛學Java的朋友們看了上面這段非常籠統的描述後仍然覺得雲裡霧裡的。

下面我們就用一個具體的例子來學習volatile的用法。

看這個例子:

public class ThreadVerify {
    public static Boolean stop = false;
    public static void main(String args[]) throws InterruptedException {
        Thread testThread = new Thread(){
            @Override
            public void run(){
                int i = 1;
                while(!stop){
                    //System.out.println("in thread: " + Thread.currentThread() + " i: " + i);
                    i++;
                }
                System.out.println("Thread stop i="+ i);
            }
        }
        ;
        testThread.start();
        Thread.sleep(1000);
        stop = true;
        System.out.println("now, in main thread stop is: " + stop);
        testThread.join();
    }
}

這段程式碼在主線程的第二行定義了一個布林變數stop, 然後主執行緒啟動一個新線程,在線程裡不停得增加計數器i的值,直到主執行緒的布林變數stop被主執行緒置為true才結束迴圈。

主執行緒用Thread.sleep停頓1秒後將布林值stop置為true。

Java volatile關鍵字的程式碼解析

因此,我們預期的結果是,上述Java程式碼執行1秒鐘後停止,並且列印出1秒鐘內計數器i的實際值。

然而,執行這個Java應用程式後,你發現它進入了死循環,在任務管理器裡發現這個Java程式CPU佔用率飆升。

原因是什麼呢?讓我們溫習下電腦專業課程作業系統中講過的記憶體模型的知識。

以Java記憶體模型為例,Java記憶體模型分為主記憶體(main memory)和工作記憶體(work memory)。主記憶體內的變數由所有執行緒共享,每個執行緒擁有自己的工作內存,裡面的變數包含了線程局部變數。主記憶體中的變數如果被執行緒使用到,則執行緒的工作記憶體會維護一份主記憶體變數的副本拷貝。

Java volatile關鍵字的程式碼解析

執行緒對變數的所有讀寫操作必須在工作記憶體中進行,不能直接操作主記憶體中的變數。不同線程之間也無法直接存取對方的工作記憶體。線程間變數的傳遞需透過主記憶體來完成。執行緒、主記憶體、工作記憶體三者之間的互動關係如下圖:

Java volatile關鍵字的程式碼解析

如果執行緒在自己的執行程式碼裡修改了定義在主執行緒(主記憶體)中的變量,修改直接發生在執行緒的工作記憶體裡,然後在某個時刻(Java程式設計師無法控制這個時刻,而是由JVM調度的),這個修改從工作記憶體寫回到主內存。

回到我們的例子。儘管主執行緒修改了stop變量,但是僅僅修改了主記憶體中的值,而操作計數器的執行緒的工作記憶體裡的stop變數還是舊的值,總是false。因此這個線程陷入了死循環。

Java volatile關鍵字的程式碼解析

知道了原理,解決方案就很簡單了。在stop變數前面加上關鍵字volatile進行修飾,這樣在計數器執行緒裡每次讀取stop的值時,volatile會強制該執行緒從主記憶體讀取,而不是從目前執行緒的工作記憶體讀取。這樣就避免了死循環。下圖顯示1秒鐘之後,計數器執行了14億次。

Java volatile關鍵字的程式碼解析

#

以上是Java volatile關鍵字的程式碼解析的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:segmentfault.com。如有侵權,請聯絡admin@php.cn刪除