鍍金池/ 問(wèn)答/Java  Linux  網(wǎng)絡(luò)安全/ 關(guān)于volatile的幾個(gè)問(wèn)題

關(guān)于volatile的幾個(gè)問(wèn)題

public class Test {
????public volatile int inc = 0;
?
????public void increase() {
????????inc++;
????}
?
????public static void main(String[] args) {
????????final Test test = new Test();
????????for(int i=0;i<10;i++){
????????????new Thread(){
????????????????public void run() {
????????????????????for(int j=0;j<1000;j++)
????????????????????????test.increase();
????????????????};
????????????}.start();
????????}
?
????????while(Thread.activeCount()>1)? //保證前面的線程都執(zhí)行完
????????????Thread.yield();
????????System.out.println(test.inc);
????} } ![clipboard.png](/img/bV6q1M)

假如某個(gè)時(shí)刻變量inc的值為10,
線程1對(duì)變量進(jìn)行自增操作,線程1先讀取了變量inc的原始值,然后線程1被阻塞了;
然后線程2對(duì)變量進(jìn)行自增操作,線程2也去讀取變量inc的原始值,由于線程1只是對(duì)變量inc進(jìn)行讀取操作,而沒(méi)有對(duì)變量進(jìn)行修改操作,所以不會(huì)導(dǎo)致線程2的工作內(nèi)存中緩存變量inc的緩存行無(wú)效,所以線程2會(huì)直接去主存讀取inc的值,發(fā)現(xiàn)inc的值時(shí)10,然后進(jìn)行加1操作,并把11寫(xiě)入工作內(nèi)存,最后寫(xiě)入主存。
然后線程1接著進(jìn)行加1操作,由于已經(jīng)讀取了inc的值,注意此時(shí)在線程1的工作內(nèi)存中inc的值仍然為10,所以線程1對(duì)inc進(jìn)行加1操作后inc的值為11,然后將11寫(xiě)入工作內(nèi)存,最后寫(xiě)入主存。

clipboard.png


**

加了volatile之后,線程2對(duì)inc進(jìn)行了更改,不會(huì)導(dǎo)致線程1要重新讀取inc嗎

**


希望能有大佬詳細(xì)講解一下 ———魯迅
回答
編輯回答
別瞎鬧

volatile僅保證可見(jiàn)性。這個(gè)可見(jiàn)性是針對(duì)讀取操作來(lái)說(shuō)的,所以你說(shuō)的情況完全有可能發(fā)生。

之所以會(huì)這樣,是因?yàn)閷?duì)個(gè)線程并發(fā)對(duì)同一個(gè)變量進(jìn)行修改時(shí),除了可見(jiàn)性,還必須保證修改過(guò)程是原子的,修改過(guò)程包括讀、自增、寫(xiě)三步。

所以你這種情況,如果把inc換成AtomicInteger就沒(méi)問(wèn)題了。

如果你的10個(gè)線程中,只有1個(gè)線程會(huì)修改inc變量,另外9個(gè)線程都只是讀取,那么就可以使用volatile,它會(huì)保證這9個(gè)線程每次讀到的都是最新的inc值。

2017年2月10日 08:57
編輯回答
枕邊人

當(dāng)讀一個(gè)volatile變量時(shí),JMM(Java內(nèi)存模型)會(huì)把該線程對(duì)應(yīng)的本地內(nèi)存置為無(wú)效。線程接下來(lái)將從主內(nèi)存中讀取共享變量。當(dāng)寫(xiě)一個(gè)volatile變量時(shí),JMM會(huì)把該線程對(duì)應(yīng)的本地內(nèi)存中的共享變量值刷新到主內(nèi)存。---<<Java并發(fā)編程的藝術(shù)>>
對(duì)于你的問(wèn)題,答案是不會(huì)。你應(yīng)該是誤解了把該線程對(duì)應(yīng)的本地內(nèi)存置為無(wú)效

2017年11月19日 09:04