java volatile案例講解
本篇來(lái)自java并發(fā)編程實(shí)戰(zhàn)關(guān)于volatile的總結(jié)。
要說(shuō)volatile,先得明白內(nèi)存可見(jiàn)性。那我們就從內(nèi)存可見(jiàn)性說(shuō)起。
一、內(nèi)存可見(jiàn)性可見(jiàn)性是一種復(fù)雜的屬性,因?yàn)榭梢?jiàn)性中的錯(cuò)誤總是會(huì)違背我們的直覺(jué)。在單線程環(huán)境中,如果向某個(gè)變量先寫(xiě)入值,然后在沒(méi)有其他寫(xiě)入操作的情況下讀取這個(gè)變量,那么總能得到相同的值。這看起來(lái)很自然。然而,當(dāng)讀操作和寫(xiě)操作在不同的線程中執(zhí)行時(shí),情況卻并非如此,這聽(tīng)起來(lái)或許有些難以接受。通常,我們無(wú)法確保執(zhí)行讀操作的線程能適時(shí)地看到其他線程寫(xiě)入的值,有時(shí)甚至是根本不可能的事情。為了確保多個(gè)想成之間對(duì)內(nèi)存寫(xiě)入操作的可見(jiàn)性,必須使用同步機(jī)制。 對(duì)于以下代碼:
public class NoVisibility { private static boolean ready; private static int number;private static class ReaderThread extends Thread{public void run(){ while(!ready)Thread.yield(); System.out.println(number);} }public static void main(String[] args){new ReaderThread().start();number = 42;ready = true; }}
NoVisibility可能會(huì)持續(xù)循環(huán)下去,因?yàn)樽x線程可能永遠(yuǎn)都看不到ready的值。一種更奇怪的現(xiàn)象是,Novisibility可能會(huì)輸出0,因?yàn)樽x線程可能看到了寫(xiě)入ready的值,但卻沒(méi)有看到之后寫(xiě)入number的值,這種現(xiàn)象被稱(chēng)為“重排序(Reordering)“。只要在某個(gè)線程中無(wú)法檢測(cè)到重排序情況,(即使在其他線程中可以很明顯地看到該線程中的重排序),那么就無(wú)法確保線程中的操作將按照程序中指定的順序來(lái)執(zhí)行。當(dāng)主線程首先寫(xiě)入number,然后在沒(méi)有同步的情況下寫(xiě)入ready,那么讀線程看到的順序可能與寫(xiě)入的順序完全相反。
在沒(méi)有同步的情況下,編譯器、處理器以及運(yùn)行時(shí)等都可能對(duì)操作的執(zhí)行順序進(jìn)行一些意想不到的調(diào)整。在缺乏足夠同步的多線程程序中,要相對(duì)內(nèi)存操作的執(zhí)行順序進(jìn)行判斷,幾乎無(wú)法得出正確的結(jié)論。
這看上去似乎是一種失敗的設(shè)計(jì),但卻能使JVM充分地利用現(xiàn)代多核處理器的強(qiáng)大性能。例如,在缺少同步的情況下,java內(nèi)存模型允許編譯器對(duì)操作順序進(jìn)行重排序,并將數(shù)值緩存在寄存器中。此外,它還允許CPU對(duì)操作順序進(jìn)行重排序,并將數(shù)值環(huán)迅在處理器特定的緩存中。
二、Volatile變量java語(yǔ)言提供了一種稍弱的同步機(jī)制,即volatile變量,用來(lái)確保將變量的更新操作通知到其他線程。當(dāng)把變量聲明為volatile類(lèi)型后,編譯器與運(yùn)行時(shí)都會(huì)注意到這個(gè)變量是共享的,因此不會(huì)將該變量上的操作和其他內(nèi)存操作一起重排序。volatile變量不會(huì)被緩存在寄存器或者對(duì)其他處理器不可見(jiàn)的地方,因此在讀取volatile類(lèi)型的變量時(shí)總會(huì)返回最新寫(xiě)入的值。
volatile與加鎖機(jī)制的區(qū)別:
加鎖機(jī)制既可以確??梢?jiàn)性又可以確保原子性,而volatile變量只能確??梢?jiàn)性。
當(dāng)且僅當(dāng)滿足以下所有條件時(shí),才應(yīng)該使用volatile變量:
對(duì)變量的寫(xiě)入操作不依賴變量的當(dāng)前值,或者你能確保只有單個(gè)線程更新變量的值。 該變量不會(huì)與其他狀態(tài)變量一起納入不變性條件中。 在訪問(wèn)變量時(shí)不需要加鎖。到此這篇關(guān)于java volatile案例講解的文章就介紹到這了,更多相關(guān)Java volatile內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!
相關(guān)文章:
1. python GUI庫(kù)圖形界面開(kāi)發(fā)之PyQt5動(dòng)態(tài)(可拖動(dòng)控件大小)布局控件QSplitter詳細(xì)使用方法與實(shí)例2. ASP將數(shù)字轉(zhuǎn)中文數(shù)字(大寫(xiě)金額)的函數(shù)3. XML 非法字符(轉(zhuǎn)義字符)4. ASP 處理JSON數(shù)據(jù)的實(shí)現(xiàn)代碼5. js開(kāi)發(fā)中的頁(yè)面、屏幕、瀏覽器的位置原理(高度寬度)說(shuō)明講解(附圖)6. CSS清除浮動(dòng)方法匯總7. 不要在HTML中濫用div8. vue跳轉(zhuǎn)頁(yè)面常用的幾種方法匯總9. CSS3實(shí)例分享之多重背景的實(shí)現(xiàn)(Multiple backgrounds)10. XML入門(mén)的常見(jiàn)問(wèn)題(三)
