1.簡介
volatile作為java中的關(guān)鍵詞之一,用以聲明變量的值可能隨時會別的線程修改,使用volatile修飾的變量會強制將修改的值立即寫入主存,主存中值的更新會使緩存中的值失效(非volatile變量不具備這樣的特性,非volatile變量的值會被緩存,線程A更新了這個值,線程B讀取這個變量的值時可能讀到的并不是是線程A更新后的值)。volatile會禁止指令重排。
2.volatile特性
volatile具有可見性、有序性,不具備原子性。
注意,volatile不具備原子性,這是volatile與java中的synchronized、java.util.concurrent.locks.Lock最大的功能差異,這一點在面試中也是非常容易問到的點。
下面來分別看下可見性、有序性、原子性:
a.原子性:如果你了解事務,那這個概念應該好理解。原子性通常指多個操作不存在只執(zhí)行一部分的情況,如果全部執(zhí)行完成那沒毛病,如果只執(zhí)行了一部分,那對不起,你得撤銷(即事務中的回滾)已經(jīng)執(zhí)行的部分。
b.可見性:當多個線程訪問同一個變量x時,線程1修改了變量x的值,線程1、線程2...線程n能夠立即讀取到線程1修改后的值。
c.有序性:即程序執(zhí)行時按照代碼書寫的先后順序執(zhí)行。在Java內(nèi)存模型中,允許編譯器和處理器對指令進行重排序,但是重排序過程不會影響到單線程程序的執(zhí)行,卻會影響到多線程并發(fā)執(zhí)行的正確性。(本文不對指令重排作介紹,但不代表它不重要,它是理解JAVA并發(fā)原理時非常重要的一個概念)。
3.volatile適用場景
a.適用于對變量的寫操作不依賴于當前值,對變量的讀取操作不依賴于非volatile變量。
b.適用于讀多寫少的場景。
c.可用作狀態(tài)標志。
d.DK中volatie應用:JDK中ConcurrentHashMap的Entry的value和next被聲明為volatile,AtomicLong中的value被聲明為volatile。AtomicLong通過CAS原理(也可以理解為樂觀鎖)保證了原子性。
4.volatile VS synchronized
volatilesynchronized修飾對象修飾變量修飾方法或代碼段可見性11有序性11原子性01線程阻塞01對比這個表格,你會不會覺得synchronized完勝volatile,答案是否定的,volatile不會讓線程阻塞,響應速度比synchronized高,這是它的優(yōu)點。