1. 程式人生 > >volatile和synchronized

volatile和synchronized

volatile 通過 syn 只有一個 存取 不能 代碼塊 多資源 obj

volatile是變量修飾符,而synchronized則是作用於代碼,方法和變量。

int i1;
int geti1() {return i1;}

volatile int i2;
int geti2() {return i2;}

int i3;
synchronized int geti3() {return i3;}

1.geti1()是得到儲存在當前線程中i1的數值,而多個線程又多份i1變量的拷貝,這些i1可能不相同。(另一個線程可能已經改變了它線程內的i1值,而這個值可以和當前線程中的i1值不相同。)

在JAVA內存模型的,有main memory(主內存區域),存放著變量當前的準確值,每個線程也有自己的memory(例如寄存器),為了性能,一個線程會在自己的memory中保存要訪問的變量的副本。這樣就會出現同一個變量在某個瞬間,在一個線程的memory中的值可能與另外一個線程memory的值,或者main memory的值不一致的情況。因此實際上存在一種可能:main memory的值i1值是1,線程1裏的i1是2,線程2裏的i1值是3,這在線程1和線程2都改變了他們各自的i1值,而且這個改變還沒來得及傳給main memory 或其他線程時就會發生。

2.geti2() 得到的是main memory的i2數值。一個變量聲明為volatile,就意味著這個變量是隨時會被其他線程修改的,因此不能將它cache在線程memory中。換句話說,一個變量經過volatile修飾後在所有線程中必須是同步的,任何線程中改變了它的值,所有其他線程立即獲得了相同的值。所以,volatile修飾的變量存取時比一般變量消耗的資源要多一點,因為線程有它自己的變量拷貝更為高效。

3.geti3()方法被synchronized修飾,用synchronized來修飾一個方法或者一個代碼的時候,能夠保證在同一時刻最多只有一個線程執行該段代碼。既然volatile關鍵字已經實現了線程間數據同步,又要synchronized幹嘛呢?當兩個並發線程訪問同一個對象object中的這個synchronized(this)同步代碼時,一個時間內只能有一個線程得到執行。另一個線程必須等待當前線程執行完這個代碼塊才能執行該代碼塊。然而,當一個線程訪問object的一個synchronized(this)同步代碼塊時,另一個線程仍然可以訪問該object中的非synchronized(this)同步代碼塊。尤其關鍵的是,當一個線程訪問object的一個synchronized(this)同步代碼塊時候,其他線程對object中所有其他synchronized(this)同步代碼塊的訪問將被阻塞。當一個線程訪問object的一個synchronized(this)同步代碼塊時,它就獲得了這個object的對象鎖,結果,其他線程對該對象所有同步代碼部分的訪問都被暫時阻塞。

總結一下區別

一,volatile是變量修飾符,而synchronized則作用於一段代碼或者方法。

二,volatile只是在線程內存和main memory(主內存)間同步某個變量的值;而synchronized通過鎖定和解鎖某個監視器同步所有變量的值。顯然synchronized要比volatile消耗更多資源。

http://www.cnblogs.com/subclass/p/7000474.html

volatile和synchronized