多線程volatile關(guān)鍵字 java里volatile關(guān)鍵字有什么特性是否能保證線程安全?
java里volatile關(guān)鍵字有什么特性是否能保證線程安全?1. Volatile不能保證原子性。簡而言之,Java有所謂的主存區(qū)和線程棧。在主內(nèi)存區(qū)域和每個線程的堆棧中都有相同變量的副本(一對多)
java里volatile關(guān)鍵字有什么特性是否能保證線程安全?
1. Volatile不能保證原子性。簡而言之,Java有所謂的主存區(qū)和線程棧。在主內(nèi)存區(qū)域和每個線程的堆棧中都有相同變量的副本(一對多)。volatile提供的可見性意味著當(dāng)每個線程訪問volatile修改的變量時,volatile確保線程可以從主存加載最新的值(相反,修改線程后同步到主存的值也應(yīng)該對其他線程可見);
2。Java的volatile的語義實(shí)際上并不涉及CPU緩存。JVM本身是一個軟件抽象,它已經(jīng)在操作系統(tǒng)之上了。由于非原子性,volatile不能保證線程安全。如果只有簡單的讀寫操作,比如set I=2,get I,就可以認(rèn)為是安全的。4Volatile被認(rèn)為比lock更輕,編程更簡單??梢允褂胿olatile的地方:對于一個變量,更新它的值不依賴于當(dāng)前值,并且該變量不會與其他變量形成一個不可變的條件。
多個線程可以讀一個變量,只有一個線程可以對這個變量進(jìn)行寫,到底要不要加鎖?
下面簡要說明以下原因:
鎖定是因?yàn)椴僮鞑皇窃拥?。讓我們用操作一來解釋它??聪旅鎯蓚€圖。
我這個操作需要
看上面的第二個圖,你能很清楚地理解這個過程嗎?
鎖定是為了確保上述三個步驟是原子操作。
回到問題上來,只有一個線程要寫,沒有競爭,所以不需要鎖定。
但是,如果你看第一張圖片,因?yàn)橹鲀?nèi)存和本地內(nèi)存的存在
在一個線程寫入后,其他線程無法立即看到它。這就是可見性問題。
添加volatile關(guān)鍵字后,它將在操作后強(qiáng)制工作內(nèi)存和主內(nèi)存同步,以確保其他線程可以立即看到它。
java volatile既然不能絕對保證線程安全,那意義何在?
確保您想要的數(shù)據(jù)是當(dāng)時的真實(shí)數(shù)據(jù)。這需要結(jié)合CPU緩存來解釋。大多數(shù)情況下,您需要的數(shù)據(jù)只是CPU緩存的數(shù)據(jù),內(nèi)存中的數(shù)據(jù)發(fā)生了變化(特別是在多核CPU的情況下)。
它可以保證訪問時數(shù)據(jù)的一致性,但不能保證處理過程中數(shù)據(jù)的一致性。