Java語(yǔ)言提供了一種稍弱的同步機(jī)制,即volatile變量,用來(lái)確保將變量的更新操作通知到其他線程。volatile變量具備兩種特性,volatile變量不會(huì)被緩存在寄存器或者對(duì)其他處理器不可見的地方,因此在讀取volatile類型的變量時(shí)總會(huì)返回最新寫入的值。
變量可見性
其一是保證該變量對(duì)所有線程可見,這里的可見性指的是當(dāng)一個(gè)線程修改了變量的值,那么新的值對(duì)于其他線程是可以立即獲取的。
禁止重排序
volatile禁止了指令重排。
它是一種比*sychronized* 更輕量級(jí)的同步鎖。我們?cè)谠L問 volatile變量時(shí)不會(huì)執(zhí)行加鎖操作,因此也就不會(huì)使執(zhí)行線程阻塞,因此volatile變量是一種比 sychronized關(guān)鍵字更輕量級(jí)的同步機(jī)制。
volatile適合這種場(chǎng)景:一個(gè)變量被多個(gè)線程共享,線程直接給這個(gè)變量賦值。當(dāng)對(duì)非volatile變量進(jìn)行讀寫的時(shí)候,每個(gè)線程先從內(nèi)存拷貝變量到CPU緩存中。如果計(jì)算機(jī)有多個(gè)CPU,每個(gè)線程可能在不同的CPU上被處理,這意味著每個(gè)線程可以拷貝到不同的CPU cache中。而聲明變量是 volatile 的,JVM 保證了每次讀變量都從內(nèi)存中讀,跳過CPU cache這一步。
適用場(chǎng)景
值得說明的是對(duì)volatile變量的單次讀/寫操作可以保證原子性的,如long和double類型變量,但是并不能保證 i++這種操作的原子性,因?yàn)楸举|(zhì)上 i++是讀、寫兩次操作。
在某些場(chǎng)景下可以代替Synchronized。但是,volatile的不能完全取代Synchronized的位置,只有在一些特殊的場(chǎng)景下,才能適用volatile。
總的來(lái)說,必須同時(shí)滿足下面兩個(gè)條件才能保證在并發(fā)環(huán)境的線程安全:
1. 對(duì)變量的寫操作不依賴于當(dāng)前值(比如 i++),或者說是單純的變量賦值(boolean flag = true)。
2. 該變量沒有包含在具有其他變量的不變式中,也就是說,不同的volatile變量之間,不能互相依賴。只有在狀態(tài)真正獨(dú)立于程序內(nèi)其他內(nèi)容時(shí)才能使用volatile。