国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

一個具體的例子學習Java volatile關鍵字

周國輝 / 1430人閱讀

摘要:下面我們就用一個具體的例子來學習的用法。主內存中的變量如果被線程使用到,則線程的工作內存會維護一份主內存變量的副本拷貝。在變量前加上關鍵字進行修飾,這樣在計數器線程里每次讀取的值時,會強制該線程從主內存讀取,而不是從當前線程的工作內存讀取。

相信大多數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代碼執行1秒鐘后停止,并且打印出1秒鐘內計數器i的實際值。

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

原因是什么呢?讓我們溫習下計算機專業課操作系統中講過的內存模型的知識。

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

線程對變量的所有讀寫操作必須在工作內存中進行,不能直接操作主內存中的變量。不同線程之間也無法直接訪問對方的工作內存。線程間變量的傳遞需通過主內存來完成。線程、主內存、工作內存三者之間的交互關系如下圖:

如果線程在自己的執行代碼里修改了定義在主線程(主內存)中的變量,修改直接發生在線程的工作內存里,然后在某個時刻(Java程序員無法控制這個時刻,而是由JVM調度的),這個修改從工作內存寫回到主內存。

回到我們的例子。盡管主線程修改了stop變量,但是僅僅修改了主內存中的值,而操作計數器的線程的工作內存里的stop變量還是舊的值,始終為false。因此這個線程陷入了死循環。

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

要獲取更多Jerry的原創技術文章,請關注公眾號"汪子熙"或者掃描下面二維碼:

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/71806.html

相關文章

  • Akka系列(四):Akka中共享內存模型

    摘要:共享內存相信對并發有所了解的同學都應該知道在推出后,對內存管理有了更高標準的規范了,這使我們開發并發程序也有更好的標準了,不會有一些模糊的定義導致的無法確定的錯誤。 通過前幾篇的學習,相信大家對Akka應該有所了解了,都說解決并發哪家強,JVM上面找Akka,那么Akka到底在解決并發問題上幫我們做了什么呢? 共享內存 眾所周知,在處理并發問題上面,最核心的一部分就是如何處理共享內存,...

    baukh789 評論0 收藏0
  • 多線程學習筆記(1):volatile和synchronized

    摘要:今天開始整理學習多線程的知識,談談最重要的兩個關鍵字和。但是這樣一個過程比較慢,在使用多線程的時候就會出現問題。有序性有序性是指多線程執行結果的正確性。這種機制在多線程中會出現問題,因此可以通過來禁止重排。 今天開始整理學習多線程的知識,談談最重要的兩個關鍵字:volatile和synchronized。 一、三個特性 1、原子性 所謂原子性操作就是指這些操作是不可中斷的,要么執行過程...

    jk_v1 評論0 收藏0
  • JAVA多線程機制解析-volatile&synchronized

    摘要:當一個線程持有重量級鎖時,另外一個線程就會被直接踢到同步隊列中等待。 java代碼先編譯成字節碼,字節碼最后編譯成cpu指令,因此Java的多線程實現最終依賴于jvm和cpu的實現 synchronized和volatile 我們先來討論一下volatile關鍵字的作用以及實現機制,每個線程看到的用volatile修飾的變量的值都是最新的,更深入的解釋就涉及到Java的內存模型了,我們...

    dendoink 評論0 收藏0
  • Java 線程和 volatile 解釋

    摘要:支持多線程,中創建線程的方式有兩種繼承類,重寫方法。多線程編程很常見的情況下是希望多個線程共享資源,通過多個線程同時消費資源來提高效率,但是新手一不小心很容易陷入一個編碼誤區。所以,在進行多線程編程的時候一定要留心多個線程是否共享資源。 文章首發于 http://jaychen.cc作者 JayChen showImg(https://segmentfault.com/img/remo...

    ytwman 評論0 收藏0
  • [Java并發-2]Java如何解決可見性問題

    摘要:誕生之處就支持多線程,所以自然有解決這些問題的辦法,而且在編程語言領域處于領先地位。,線程規則這條是關于線程啟動的。在語言里面,的語義本質上是一種可見性,意味著事件對事件來說是可見的,無論事件和事件是否發生在同一個線程里。 之前我們說了:1,可見性2,原子性3,有序性3個并發BUG的之源,這三個也是編程領域的共性問題。Java誕生之處就支持多線程,所以自然有解決這些問題的辦法,而且在編...

    lk20150415 評論0 收藏0

發表評論

0條評論

周國輝

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<