2018-java必備的面試-volatile/synchronized/wait()/sleep()

1、在Java中 wait()和sleep()方法的不同?

最大的不同是在等待時wait()會釋放鎖,而sleep()會一直持有鎖。wait()通常被用於線程間的交互,sleep()通常被用於暫停執行。

2、synchronized和volatile關鍵字的作用

一旦一個共享變量(類的成員變量、類的靜態成員變量)被volatile修飾之後,那麼就具備了兩層語義:

  1. 保證了不同線程對這個變量進行操作時的可見性,即一個線程修改了某個變量的值,這個新值對其他線程來說是立即可見的。
  2. 禁止進行指令重排序。volatile本質是在告訴jvm當前變量在寄存器(工作內存)中的值是不確定的,需要從主存中讀取;synchronized則是鎖定當前變量,只有當前線程可以訪問該變量,其他線程被阻塞住。
  • volatile僅能使用在變量級別;synchronized則可以使用在變量、方法和類級別;
  • volatile僅能實現變量的修改可見性,並不能保證原子性;synchronized則可以保證變量修改的可見性和原子性;
  • volatile不會造成線程阻塞;synchronized可能會造成線程阻塞;
  • volatile標記的變量不會被編譯器優化;synchronized標記的變量可以被編譯器優化。

3、分析如下代碼執行完之後輸出的結果確定為1000嗎?原因是什麼?(由於是分析代碼,所以不給註釋)

2018-java必備的面試-volatile/synchronized/wait()/sleep()

分析代碼

答案是不一定,或者不等於1000。原因:在java的內存模型中每一個線程運行時都有一個線程,線程棧保存了線程運行時變量的信息。當線程訪問某一個對象值的時候,首先通過對象的引用找到對應在 堆內存的變量的值,然後把堆內存變量的具體load到線程本地內存中,建立一個副本,之後線程就不再和對象在堆內存變量值有任何聯繫,而是直接修改副本變量的值,在修改完之後的某一時刻(線程退出之前),自動把線程副本的值寫到對象在堆中的變量。這樣在堆中對象的值就產生變化了。也就是說上面主函數中開啟1000個子線程,每個線程都會有一個變量副本,每個線程修改變量都只是臨時修改了自己的副本,當線程結束時再將修改的值寫入主內存中,這樣就出現了線程安全問題。因此結果就不可能等於1000,一般都會小於1000。上面的解釋用如下圖表示:

2018-java必備的面試-volatile/synchronized/wait()/sleep()

2018-java必備的面試-volatile/synchronized/wait()/sleep()


分享到:


相關文章: