摘要:下面我們就用一個(gè)具體的例子來(lái)學(xué)習(xí)的用法。主內(nèi)存中的變量如果被線程使用到,則線程的工作內(nèi)存會(huì)維護(hù)一份主內(nèi)存變量的副本拷貝。在變量前加上關(guān)鍵字進(jìn)行修飾,這樣在計(jì)數(shù)器線程里每次讀取的值時(shí),會(huì)強(qiáng)制該線程從主內(nèi)存讀取,而不是從當(dāng)前線程的工作內(nèi)存讀取。
相信大多數(shù)Java程序員都學(xué)習(xí)過(guò)volatile這個(gè)關(guān)鍵字的用法。百度百科上對(duì)volatile的定義:
volatile是一個(gè)類型修飾符(type specifier),被設(shè)計(jì)用來(lái)修飾被不同線程訪問(wèn)和修改的變量。volatile的作用是作為指令關(guān)鍵字,確保本條指令不會(huì)因編譯器的優(yōu)化而省略,且要求每次直接讀值。
可能有很多剛學(xué)Java的朋友們看了上面這段非常籠統(tǒng)的描述后仍然覺(jué)得云里霧里的。
下面我們就用一個(gè)具體的例子來(lái)學(xué)習(xí)volatile的用法。
看這個(gè)例子:
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(); } }
這段代碼在主線程的第二行定義了一個(gè)布爾變量stop, 然后主線程啟動(dòng)一個(gè)新線程,在線程里不停得增加計(jì)數(shù)器i的值,直到主線程的布爾變量stop被主線程置為true才結(jié)束循環(huán)。
主線程用Thread.sleep停頓1秒后將布爾值stop置為true。
因此,我們期望的結(jié)果是,上述Java代碼執(zhí)行1秒鐘后停止,并且打印出1秒鐘內(nèi)計(jì)數(shù)器i的實(shí)際值。
然而,執(zhí)行這個(gè)Java應(yīng)用后,你發(fā)現(xiàn)它進(jìn)入了死循環(huán),在任務(wù)管理器里發(fā)現(xiàn)這個(gè)Java程序CPU占用率飆升。
原因是什么呢?讓我們溫習(xí)下計(jì)算機(jī)專業(yè)課操作系統(tǒng)中講過(guò)的內(nèi)存模型的知識(shí)。
以Java內(nèi)存模型為例,Java內(nèi)存模型分為主內(nèi)存(main memory)和工作內(nèi)存(work memory)。主內(nèi)存內(nèi)的變量由所有線程共享,每個(gè)線程擁有自己的工作內(nèi)存,里面的變量包含了線程局部變量。主內(nèi)存中的變量如果被線程使用到,則線程的工作內(nèi)存會(huì)維護(hù)一份主內(nèi)存變量的副本拷貝。
線程對(duì)變量的所有讀寫(xiě)操作必須在工作內(nèi)存中進(jìn)行,不能直接操作主內(nèi)存中的變量。不同線程之間也無(wú)法直接訪問(wèn)對(duì)方的工作內(nèi)存。線程間變量的傳遞需通過(guò)主內(nèi)存來(lái)完成。線程、主內(nèi)存、工作內(nèi)存三者之間的交互關(guān)系如下圖:
如果線程在自己的執(zhí)行代碼里修改了定義在主線程(主內(nèi)存)中的變量,修改直接發(fā)生在線程的工作內(nèi)存里,然后在某個(gè)時(shí)刻(Java程序員無(wú)法控制這個(gè)時(shí)刻,而是由JVM調(diào)度的),這個(gè)修改從工作內(nèi)存寫(xiě)回到主內(nèi)存。
回到我們的例子。盡管主線程修改了stop變量,但是僅僅修改了主內(nèi)存中的值,而操作計(jì)數(shù)器的線程的工作內(nèi)存里的stop變量還是舊的值,始終為false。因此這個(gè)線程陷入了死循環(huán)。
知道了原理,解決方案就很簡(jiǎn)單了。在stop變量前加上關(guān)鍵字volatile進(jìn)行修飾,這樣在計(jì)數(shù)器線程里每次讀取stop的值時(shí),volatile會(huì)強(qiáng)制該線程從主內(nèi)存讀取,而不是從當(dāng)前線程的工作內(nèi)存讀取。這樣就避免了死循環(huán)。下圖顯示1秒鐘之后,計(jì)數(shù)器執(zhí)行了14億次。
要獲取更多Jerry的原創(chuàng)技術(shù)文章,請(qǐng)關(guān)注公眾號(hào)"汪子熙"或者掃描下面二維碼:
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://www.ezyhdfw.cn/yun/71806.html
摘要:共享內(nèi)存相信對(duì)并發(fā)有所了解的同學(xué)都應(yīng)該知道在推出后,對(duì)內(nèi)存管理有了更高標(biāo)準(zhǔn)的規(guī)范了,這使我們開(kāi)發(fā)并發(fā)程序也有更好的標(biāo)準(zhǔn)了,不會(huì)有一些模糊的定義導(dǎo)致的無(wú)法確定的錯(cuò)誤。 通過(guò)前幾篇的學(xué)習(xí),相信大家對(duì)Akka應(yīng)該有所了解了,都說(shuō)解決并發(fā)哪家強(qiáng),JVM上面找Akka,那么Akka到底在解決并發(fā)問(wèn)題上幫我們做了什么呢? 共享內(nèi)存 眾所周知,在處理并發(fā)問(wèn)題上面,最核心的一部分就是如何處理共享內(nèi)存,...
摘要:今天開(kāi)始整理學(xué)習(xí)多線程的知識(shí),談?wù)勛钪匾膬蓚€(gè)關(guān)鍵字和。但是這樣一個(gè)過(guò)程比較慢,在使用多線程的時(shí)候就會(huì)出現(xiàn)問(wèn)題。有序性有序性是指多線程執(zhí)行結(jié)果的正確性。這種機(jī)制在多線程中會(huì)出現(xiàn)問(wèn)題,因此可以通過(guò)來(lái)禁止重排。 今天開(kāi)始整理學(xué)習(xí)多線程的知識(shí),談?wù)勛钪匾膬蓚€(gè)關(guān)鍵字:volatile和synchronized。 一、三個(gè)特性 1、原子性 所謂原子性操作就是指這些操作是不可中斷的,要么執(zhí)行過(guò)程...
摘要:當(dāng)一個(gè)線程持有重量級(jí)鎖時(shí),另外一個(gè)線程就會(huì)被直接踢到同步隊(duì)列中等待。 java代碼先編譯成字節(jié)碼,字節(jié)碼最后編譯成cpu指令,因此Java的多線程實(shí)現(xiàn)最終依賴于jvm和cpu的實(shí)現(xiàn) synchronized和volatile 我們先來(lái)討論一下volatile關(guān)鍵字的作用以及實(shí)現(xiàn)機(jī)制,每個(gè)線程看到的用volatile修飾的變量的值都是最新的,更深入的解釋就涉及到Java的內(nèi)存模型了,我們...
摘要:支持多線程,中創(chuàng)建線程的方式有兩種繼承類,重寫(xiě)方法。多線程編程很常見(jiàn)的情況下是希望多個(gè)線程共享資源,通過(guò)多個(gè)線程同時(shí)消費(fèi)資源來(lái)提高效率,但是新手一不小心很容易陷入一個(gè)編碼誤區(qū)。所以,在進(jìn)行多線程編程的時(shí)候一定要留心多個(gè)線程是否共享資源。 文章首發(fā)于 http://jaychen.cc作者 JayChen showImg(https://segmentfault.com/img/remo...
摘要:誕生之處就支持多線程,所以自然有解決這些問(wèn)題的辦法,而且在編程語(yǔ)言領(lǐng)域處于領(lǐng)先地位。,線程規(guī)則這條是關(guān)于線程啟動(dòng)的。在語(yǔ)言里面,的語(yǔ)義本質(zhì)上是一種可見(jiàn)性,意味著事件對(duì)事件來(lái)說(shuō)是可見(jiàn)的,無(wú)論事件和事件是否發(fā)生在同一個(gè)線程里。 之前我們說(shuō)了:1,可見(jiàn)性2,原子性3,有序性3個(gè)并發(fā)BUG的之源,這三個(gè)也是編程領(lǐng)域的共性問(wèn)題。Java誕生之處就支持多線程,所以自然有解決這些問(wèn)題的辦法,而且在編...
閱讀 2489·2019-08-30 15:44
閱讀 1436·2019-08-30 13:01
閱讀 3459·2019-08-30 11:22
閱讀 3246·2019-08-29 15:23
閱讀 1776·2019-08-29 12:22
閱讀 3525·2019-08-26 13:58
閱讀 3589·2019-08-26 12:17
閱讀 3634·2019-08-26 12:16