自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

圖解:volatile 和原子類的異同

開發(fā) 前端
對(duì)于會(huì)被多個(gè)線程同時(shí)操作的計(jì)數(shù)器 Counter 的場景,這種場景的一個(gè)典型特點(diǎn)就是,它不僅僅是一個(gè)簡單的賦值操作,而是需要先讀取當(dāng)前的值,然后在此基礎(chǔ)上進(jìn)行一定的修改,再把它給賦值回去。這樣一來,我們的 volatile 就不足以保證這種情況的線程安全了。
本文轉(zhuǎn)載自微信公眾號(hào)「JerryCodes」,作者 KyleJerry 。轉(zhuǎn)載本文請聯(lián)系JerryCodes公眾號(hào)。
  • volatile和原子類
  • 原子類和 volatile 的使用場景
  • 總結(jié)

volatile和原子類

我們首先看一個(gè)案例。如圖所示,我們有兩個(gè)線程。

 

在圖中左上角可以看出,有一個(gè)公共的 boolean flag 標(biāo)記位,最開始賦值為 true。

然后線程 2 會(huì)進(jìn)入一個(gè) while 循環(huán),并且根據(jù)這個(gè) flag 也就是標(biāo)記位的值來決定是否繼續(xù)執(zhí)行或著退出。

最開始由于 flag 的值是 true,所以首先會(huì)在這里執(zhí)行一定時(shí)期的循環(huán)。然后假設(shè)在某一時(shí)刻,線程 1 把這個(gè) flag 的值改為 false 了,它所希望的是,線程 2 看到這個(gè)變化后停止運(yùn)行。

但是這樣做其實(shí)是有風(fēng)險(xiǎn)的,線程 2 可能并不能立刻停下來,也有可能過一段時(shí)間才會(huì)停止,甚至在最極端的情況下可能永遠(yuǎn)都不會(huì)停止。

為了理解發(fā)生這種情況的原因,我們首先來看一下 CPU 的內(nèi)存結(jié)構(gòu),這里是一個(gè)雙核的 CPU 的簡單示意圖:

 

可以看出,線程 1 和線程 2 分別在不同的 CPU 核心上運(yùn)行,每一個(gè)核心都有自己的本地內(nèi)存,并且在下方也有它們共享的內(nèi)存。

最開始它們都可以讀取到 flag 為 true ,不過當(dāng)線程 1 這個(gè)值改為 false 之后,線程 2 并不能及時(shí)看到這次修改,因?yàn)榫€程 2 不能直接訪問線程 1 的本地內(nèi)存,這樣的問題就是一個(gè)非常典型的可見性問題。

 

要想解決這個(gè)問題,我們只需要在變量的前面加上 volatile 關(guān)鍵字修飾,只要我們加上這個(gè)關(guān)鍵字,那么每一次變量被修改的時(shí)候,其他線程對(duì)此都可見,這樣一旦線程 1 改變了這個(gè)值,那么線程 2 就可以立刻看到,因此就可以退出 while 循環(huán)了。

 

之所以加了關(guān)鍵字之后就就可以讓它擁有可見性,原因在于有了這個(gè)關(guān)鍵字之后,線程 1 的更改會(huì)被 flush 到共享內(nèi)存中,然后又會(huì)被 refresh 到線程 2 的本地內(nèi)存中,這樣線程 2 就能感受到這個(gè)變化了,所以 volatile 這個(gè)關(guān)鍵字最主要是用來解決可見性問題的,可以一定程度上保證線程安全。

現(xiàn)在讓我們回顧一下很熟悉的多線程同時(shí)進(jìn)行 value++ 的場景,如圖所示:

 

如果它被初始化為每個(gè)線程都加 1000 次,最終的結(jié)果很可能不是 2000。由于 value++ 不是原子的,所以在多線程的情況下,會(huì)出現(xiàn)線程安全問題。但是如果我們在這里使用 volatile 關(guān)鍵字,能不能解決問題呢?

 

很遺憾,即便使用了 volatile 也是不能保證線程安全的,因?yàn)檫@里的問題不單單是可見性問題,還包含原子性問題。

我們有多種辦法可以解決這里的問題,第 1 種是使用synchronized 關(guān)鍵字,如圖所示:


 

 

這樣一來,兩個(gè)線程就不能同時(shí)去更改 value 的數(shù)值,保證了 value++ 語句的原子性,并且 synchronized 同樣保證了可見性,也就是說,當(dāng)?shù)?1 個(gè)線程修改了 value 值之后,第 2 個(gè)線程可以立刻看見本次修改的結(jié)果。

解決這個(gè)問題的第 2 個(gè)方法,就是使用我們的原子類,如圖所示:

 

比如用一個(gè) AtomicInteger,然后每個(gè)線程都調(diào)用它的 incrementAndGet 方法。

在利用了原子變量之后就無需加鎖,我們可以使用它的 incrementAndGet 方法,這個(gè)操作底層由 CPU 指令保證原子性,所以即便是多個(gè)線程同時(shí)運(yùn)行,也不會(huì)發(fā)生線程安全問題。

原子類和 volatile 的使用場景

我們可以看出,volatile 和原子類的使用場景是不一樣的,如果我們有一個(gè)可見性問題,那么可以使用 volatile 關(guān)鍵字,但如果我們的問題是一個(gè)組合操作,需要用同步來解決原子性問題的話,那么可以使用原子變量,而不能使用 volatile 關(guān)鍵字。

通常情況下,volatile 可以用來修飾 boolean 類型的標(biāo)記位,因?yàn)閷?duì)于標(biāo)記位來講,直接的賦值操作本身就是具備原子性的,再加上 volatile 保證了可見性,那么就是線程安全的了。

 

總結(jié)

對(duì)于會(huì)被多個(gè)線程同時(shí)操作的計(jì)數(shù)器 Counter 的場景,這種場景的一個(gè)典型特點(diǎn)就是,它不僅僅是一個(gè)簡單的賦值操作,而是需要先讀取當(dāng)前的值,然后在此基礎(chǔ)上進(jìn)行一定的修改,再把它給賦值回去。這樣一來,我們的 volatile 就不足以保證這種情況的線程安全了。我們需要使用原子類來保證線程安全。

 

責(zé)任編輯:武曉燕 來源: JerryCodes
相關(guān)推薦

2020-12-11 11:11:44

原子類JavaCAS

2022-12-06 08:42:28

2022-08-17 07:53:10

Volatile關(guān)鍵字原子性

2024-08-09 08:41:14

2021-06-07 17:12:22

線程安全Atomic

2023-12-01 08:54:50

Java原子類型

2024-11-21 14:55:37

2023-12-14 07:36:16

Java并發(fā)原子類

2009-12-29 15:56:57

2024-03-15 08:18:25

volatileAtomic關(guān)鍵字

2024-08-14 18:18:47

2015-05-22 09:49:25

2009-12-18 15:23:03

Vista和XP路由設(shè)

2009-11-06 10:11:34

WCF和Web Ser

2024-10-16 08:26:42

2011-04-02 13:35:21

多線程編程多線程java

2011-05-26 15:33:29

volatilejava

2021-09-02 13:38:48

Eslint Babel 插件

2016-01-25 10:48:15

大數(shù)據(jù)框架HadoopSpark

2024-01-08 09:36:47

管理庫代碼
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)