淺談Java內(nèi)存模型之happens-before
happens-before原則非常重要,它是判斷數(shù)據(jù)是否存在競(jìng)爭(zhēng)、線程是否安全的主要依據(jù),依靠這個(gè)原則,我們解決在并發(fā)環(huán)境下兩操作之間是否可能存在沖突的所有問(wèn)題。下面我們就一個(gè)簡(jiǎn)單的例子稍微了解下happens-before ;
i = 1; //線程A執(zhí)行
j = i ; //線程B執(zhí)行
j 是否等于1呢?假定線程A的操作(i = 1)happens-before線程B的操作(j = i),那么可以確定線程B執(zhí)行后j = 1 一定成立,如果他們不存在happens-before原則,那么j = 1 不一定成立。這就是happens-before原則的威力。
happens-before原則定義如下:
1. 如果一個(gè)操作happens-before另一個(gè)操作,那么第一個(gè)操作的執(zhí)行結(jié)果將對(duì)第二個(gè)操作可見(jiàn),而且第一個(gè)操作的執(zhí)行順序排在第二個(gè)操作之前。
2. 兩個(gè)操作之間存在happens-before關(guān)系,并不意味著一定要按照happens-before原則制定的順序來(lái)執(zhí)行。如果重排序之后的執(zhí)行結(jié)果與按照happens-before關(guān)系來(lái)執(zhí)行的結(jié)果一致,那么這種重排序并不非法。
下面是happens-before原則規(guī)則:
1.程序次序規(guī)則:一個(gè)線程內(nèi),按照代碼順序,書(shū)寫(xiě)在前面的操作先行發(fā)生于書(shū)寫(xiě)在后面的操作;
2.鎖定規(guī)則:一個(gè)unLock操作先行發(fā)生于后面對(duì)同一個(gè)鎖額lock操作;
3.volatile變量規(guī)則:對(duì)一個(gè)變量的寫(xiě)操作先行發(fā)生于后面對(duì)這個(gè)變量的讀操作;
4.傳遞規(guī)則:如果操作A先行發(fā)生于操作B,而操作B又先行發(fā)生于操作C,則可以得出操作A先行發(fā)生于操作C;
5.線程啟動(dòng)規(guī)則:Thread對(duì)象的start()方法先行發(fā)生于此線程的每個(gè)一個(gè)動(dòng)作;
6.線程中斷規(guī)則:對(duì)線程interrupt()方法的調(diào)用先行發(fā)生于被中斷線程的代碼檢測(cè)到中斷事件的發(fā)生;
7.線程終結(jié)規(guī)則:線程中所有的操作都先行發(fā)生于線程的終止檢測(cè),我們可以通過(guò)Thread.join()方法結(jié)束、Thread.isAlive()的返回值手段檢測(cè)到線程已經(jīng)終止執(zhí)行;
8.對(duì)象終結(jié)規(guī)則:一個(gè)對(duì)象的初始化完成先行發(fā)生于他的finalize()方法的開(kāi)始;
我們來(lái)詳細(xì)看看上面每條規(guī)則(摘自《深入理解Java虛擬機(jī)第12章》):
- 程序次序規(guī)則:一段代碼在單線程中執(zhí)行的結(jié)果是有序的。注意是執(zhí)行結(jié)果,因?yàn)樘摂M機(jī)、處理器會(huì)對(duì)指令進(jìn)行重排序(重排序后面會(huì)詳細(xì)介紹)。雖然重排序了,但是并不會(huì)影響程序的執(zhí)行結(jié)果,所以程序最終執(zhí)行的結(jié)果與順序執(zhí)行的結(jié)果是一致的。故而這個(gè)規(guī)則只對(duì)單線程有效,在多線程環(huán)境下無(wú)法保證正確性。
- volatile變量規(guī)則:這是一條比較重要的規(guī)則,它標(biāo)志著volatile保證了線程可見(jiàn)性。通俗點(diǎn)講就是如果一個(gè)線程先去寫(xiě)一個(gè)volatile變量,然后一個(gè)線程去讀這個(gè)變量,那么這個(gè)寫(xiě)操作一定是happens-before讀操作的。
- 傳遞規(guī)則:提現(xiàn)了happens-before原則具有傳遞性,即A happens-before B , B happens-before C,那么A happens-before C
- 線程啟動(dòng)規(guī)則:假定線程A在執(zhí)行過(guò)程中,通過(guò)執(zhí)行ThreadB.start()來(lái)啟動(dòng)線程B,那么線程A對(duì)共享變量的修改在接下來(lái)線程B開(kāi)始執(zhí)行后確保對(duì)線程B可見(jiàn)。
- 線程終結(jié)規(guī)則:假定線程A在執(zhí)行的過(guò)程中,通過(guò)制定ThreadB.join()等待線程B終止,那么線程B在終止之前對(duì)共享變量的修改在線程A等待返回后可見(jiàn)。
上面八條是原生Java滿(mǎn)足Happens-before關(guān)系的規(guī)則,但是我們可以對(duì)他們進(jìn)行推導(dǎo)出其他滿(mǎn)足happens-before的規(guī)則:
1.將一個(gè)元素放入一個(gè)線程安全的隊(duì)列的操作Happens-Before從隊(duì)列中取出這個(gè)元素的操作
2.將一個(gè)元素放入一個(gè)線程安全容器的操作Happens-Before從容器中取出這個(gè)元素的操作
3.在CountDownLatch上的倒數(shù)操作Happens-Before CountDownLatch#await()操作
4.釋放Semaphore許可的操作Happens-Before獲得許可操作
5.Future表示的任務(wù)的所有操作Happens-Before Future#get()操作
6.向Executor提交一個(gè)Runnable或Callable的操作Happens-Before任務(wù)開(kāi)始執(zhí)行操作
這里再說(shuō)一遍happens-before的概念:如果兩個(gè)操作不存在上述(前面8條 + 后面6條)任一一個(gè)happens-before規(guī)則,那么這兩個(gè)操作就沒(méi)有順序的保障,JVM可以對(duì)這兩個(gè)操作進(jìn)行重排序。如果操作A happens-before操作B,那么操作A在內(nèi)存上所做的操作對(duì)操作B都是可見(jiàn)的。
下面就用一個(gè)簡(jiǎn)單的例子來(lái)描述下happens-before原則:
private int i = 0; public void write(int j ){ i = j; } public int read(){ return i; }
我們約定線程A執(zhí)行write(),線程B執(zhí)行read(),且線程A優(yōu)先于線程B執(zhí)行,那么線程B獲得結(jié)果是什么?;我們就這段簡(jiǎn)單的代碼一次分析happens-before的規(guī)則(規(guī)則5、6、7、8 + 推導(dǎo)的6條可以忽略,因?yàn)樗麄兒瓦@段代碼毫無(wú)關(guān)系):
1.由于兩個(gè)方法是由不同的線程調(diào)用,所以肯定不滿(mǎn)足程序次序規(guī)則;
2.兩個(gè)方法都沒(méi)有使用鎖,所以不滿(mǎn)足鎖定規(guī)則;
3.變量i不是用volatile修飾的,所以volatile變量規(guī)則不滿(mǎn)足;
4.傳遞規(guī)則肯定不滿(mǎn)足;
所以我們無(wú)法通過(guò)happens-before原則推導(dǎo)出線程A happens-before線程B,雖然可以確認(rèn)在時(shí)間上線程A優(yōu)先于線程B指定,但是就是無(wú)法確認(rèn)線程B獲得的結(jié)果是什么,所以這段代碼不是線程安全的。那么怎么修復(fù)這段代碼呢?滿(mǎn)足規(guī)則2、3任一即可。
happen-before原則是JMM中非常重要的原則,它是判斷數(shù)據(jù)是否存在競(jìng)爭(zhēng)、線程是否安全的主要依據(jù),保證了多線程環(huán)境下的可見(jiàn)性。
下圖是happens-before與JMM的關(guān)系圖(摘自《Java并發(fā)編程的藝術(shù)》)
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家
相關(guān)文章
Java線程的創(chuàng)建介紹及實(shí)現(xiàn)方式示例
這篇文章主要為大家介紹了Java線程的創(chuàng)建介紹及實(shí)現(xiàn)方式示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09Spring Boot 項(xiàng)目啟動(dòng)自動(dòng)執(zhí)行方法的兩種實(shí)現(xiàn)方式
這篇文章主要介紹了Spring Boot 項(xiàng)目啟動(dòng)自動(dòng)執(zhí)行方法的兩種實(shí)現(xiàn)方式,幫助大家更好的理解和學(xué)習(xí)使用Spring Boot框架,感興趣的朋友可以了解下2021-05-05spring自定義注解實(shí)現(xiàn)攔截器的實(shí)現(xiàn)方法
本篇文章主要介紹了spring自定義注解實(shí)現(xiàn)攔截器的實(shí)現(xiàn)方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-08-08Spring Cloud 系列之服務(wù)調(diào)用 OpenFeign的實(shí)現(xiàn)
這篇文章主要介紹了Spring Cloud 系列之服務(wù)調(diào)用 OpenFeign的實(shí)現(xiàn),需要的朋友可以參考下2020-11-11Java8新特性O(shè)ptional類(lèi)及新時(shí)間日期API示例詳解
這篇文章主要為大家介紹了Java8新特性O(shè)ptional類(lèi)及新時(shí)間日期API示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11Java字節(jié)緩存流的構(gòu)造方法之文件IO流
這篇文章主要介紹了Java字節(jié)緩存流的構(gòu)造方法之文件IO流,同時(shí)也介紹了字符流中的一些相關(guān)的內(nèi)容,并且通過(guò)大量的案例供大家理解。最后通過(guò)一些經(jīng)典的案例幫助大家對(duì)前面所學(xué)的知識(shí)做了一個(gè)綜合的應(yīng)用,需要的朋友可以參考一下2022-04-04Spring中的@ConfigurationProperties詳解
這篇文章主要介紹了Spring中的@ConfigurationProperties詳解,ConfigurationProperties注解主要用于將外部配置文件配置的屬性填充到這個(gè)Spring Bean實(shí)例中,需要的朋友可以參考下2023-09-09springboot 配置DRUID數(shù)據(jù)源的方法實(shí)例分析
這篇文章主要介紹了springboot 配置DRUID數(shù)據(jù)源的方法,結(jié)合實(shí)例形式分析了springboot 配置阿里DRUID數(shù)據(jù)源的具體步驟與相關(guān)操作技巧,需要的朋友可以參考下2019-12-12