java Volatile與Synchronized的區別
在研究并發程序時,我們可能都知道volatile和synchronized是用于多線程中,用于線程安全和變量可見性的,但是具體兩者怎么使用,有何區別可能還是稀里糊涂一知半解,在此就自己簡單的理解總結一下二者的區別,和大家一塊兒學習!我們需要了解java中關鍵字volatile和synchronized關鍵字的使用以及lock類的用法。
首先,了解下java的內存模型:
java的線程內存模型中定義了每個線程都有一份自己的共享變量副本(本地內存),里面存放自己私有的數據,其他線程不能直接訪問,而一些共享變量則存在主內存中,供所有線程訪問。上圖中,如果線程A和線程B要進行通信,就要經過主內存,比如線程B要獲取線程A修改后的共享變量的值,要經過下面兩步: (1)、線程A修改自己的共享變量副本,并刷新到了主內存中。 (2)、線程B讀取主內存中被A更新過的共享變量的值,同步到自己的共享變量副本中。總結:在java內存模型中,共享變量存放在主內存中,每個線程都有自己的本地內存,當多個線程同時訪問一個數據的時候,可能本地內存沒有及時刷新到主內存,所以就會發生線程安全問題。
java多線程中的三個特性: 原子性:即一個操作或者多個操作 要么全部執行并且執行的過程不會被任何因素打斷,要么就都不執行。一個很經典的例子就是銀行賬戶轉賬問題:比如從賬戶A向賬戶B轉1000元,那么必然包括2個操作:從賬戶A減去1000元,往賬戶B加上1000元。這2個操作必須要具備原子性才能保證不出現一些意外的問題。 可見性:當多個線程訪問同一個變量時,一個線程修改了這個變量的值,其他線程能夠立即看得到修改的值。 有序性:就是程序執行的順序按照代碼的先后順序執行。一般來說處理器為了提高程序運行效率,可能會對輸入代碼進行優化,它不保證程序中各個語句的執行先后順序同代碼中的順序一致,但是它會保證程序最終執行結果和代碼順序執行的結果是一致的。如下:int a = 20; //語句1int r = 3; //語句2a = a + 5; //語句3r = a*a; //語句4
因為重排序,他的執行順序還可能為 2 -1 - 3 - 4,1 - 3 - 2 - 4。但絕不可能 2 -1 - 4 - 3,因為這打破了依賴關系。顯然重排序對單線程運行是不會有任何問題,而多線程就不一定了,所以我們在多線程編程時就得考慮這個問題了。
Volatile關鍵字的作用:其實volatile關鍵字的作用就是保證了可見性和有序性(不保證原子性),如果一個共享變量被volatile關鍵字修飾,那么如果一個線程修改了這個共享變量后,其他線程是立馬可知的。如果線程A修改了自己的共享變量副本,這時如果該共享變量沒有被volatile修飾,那么本次修改不一定會馬上將修改結果刷新到主存中,如果此時B去主存中讀取共享變量的值,那么這個值就是沒有被A修改之前的值。如果該共享變量被volatile修飾了,那么本次修改結果會強制立刻刷新到主存中,如果此時B去主存中讀取共享變量的值,那么這個值就是被A修改之后的值了。volatile禁止指令重排序優化,在指令重排序優化時,在volatile變量之前的指令不能在volatile之后執行,在volatile之后的指令也不能在volatile之前執行,所以它保證了有序性。volatile 的讀性能消耗與普通變量幾乎相同,但是寫操作稍慢,因為它需要在本地代碼中插入許多內存屏障指令(是一種CPU指令,用于控制特定條件下的重排序和內存可見性問題。Java編譯器也會根據內存屏障的規則禁止重排序。)來保證處理器不發生亂序執行。
Synchronized關鍵字的作用:synchronized提供了同步鎖的概念,被synchronized修飾的代碼段可以防止被多個線程同時執行,必須一個線程把synchronized修飾的代碼段都執行完畢了,其他的線程才能開始執行這段代碼。 因為synchronized保證了在同一時刻,只能有一個線程執行同步代碼塊,所以執行同步代碼塊的時候相當于是單線程操作了,那么線程的可見性、原子性、有序性(線程之間的執行順序)它都能保證了。synchronized并沒有禁止重排序,但是synchronized相當于是一個單線程了,所以有沒有重排序對程序都是沒有影響的。
Volatile和synchronized的區別:(1)、volatile只能作用于變量,使用范圍較小。synchronized可以用在變量、方法、類、同步代碼塊等,使用范圍比較廣。(2)、volatile只能保證可見性和有序性,不能保證原子性。而可見性、有序性、原子性synchronized都可以包證。(3)、volatile不會造成線程阻塞。synchronized可能會造成線程阻塞。(4)、在性能方面synchronized關鍵字是防止多個線程同時執行一段代碼,就會影響程序執行效率,而volatile關鍵字在某些情況下性能要優于synchronized。
什么是重排序:重排序是指編譯器和處理器為了優化程序性能而對指令序列進行重新排序的一種手段。但是重排序可以保證最終執行的結果是與程序順序執行的結果一致,并且只會對不存在數據依賴性的指令進行重排序,這個重排序在單線程下對最終執行結果是沒有影響的,但是在多線程下就會存在問題。
可以看一個例子:class TestExample { int a = 0; boolean flag = false; // 寫入的線程 public void writer() { a = 1; // 1 flag = true; // 2 } //讀取的線程public void reader() { if (flag) { // 3 int i = a * a; // 4 }}}
如上面代碼,如果兩個線程同時執行在沒有發生重排序的時候int i =1,如果發生了重排序那么1,2的位置因為不存在數據依賴可以會發生位置的互換。那么這時候int i =0;當然這個在單線程是沒有問題的。只有在多線程才會發生這種情況
volatile int a = 0; volatile boolean flag = false;
我們只需要加上volatile關鍵字也是可以避免這種問題的,volatile是禁止重排序的。
什么是數據依賴?int a = 1;(1) int b = 2;(2) int c= a + b;(3)
這里面第三步就存在數據依賴 (依賴a和b的值)。編譯器和處理器在重排序時,會遵守數據依賴性,編譯器和處理器不會改變存在數據依賴關系的兩個操作的執行順序。所以這里面無論步驟(1)(2)有沒有發生重排序,步驟(3)都是在他們之后執行。這里所說的數據依賴性僅針對單個處理器中執行的指令序列和單個線程中執行的操作,不同處理器之間和不同線程之間的數據依賴性不被編譯器和處理器考慮。
以上就是java Volatile與Synchronized的區別的詳細內容,更多關于java Volatile與Synchronized區別的資料請關注好吧啦網其它相關文章!
相關文章:
