面試問題:
下面的方法是否線程安全?怎樣讓它成為線程安全的方法?
class MyCounter { private static int counter = 0; public static int getCount() { return counter++; } }
本篇文章將解釋一個常見的面試題,該問題被谷歌和很多其它公司問起過。它涉及的相對比較初級,而不是關於怎樣去設計復雜的並發程序。
首先,這個問題的答案是No,因為counter++操作不是一個原子操作,而是由多個原子操作組成。
舉個例子,在如下情況:一個線程正在訪問該數據,另一個線程正在執行遞增操作;
當線程Thread 1在t1時刻訪問該方法,線程Thread 2有可能還沒執行完這個方法的操作。因此,返回線程Thread 1的值有可能還沒被遞增過。
使用關鍵字synchronized修飾getCount方法可以使它線程安全。當使用synchronized修飾靜態方法,該類對象成為了鎖。
使用synchronized就足夠了嗎,答案是Yes.
class MyCounter { private static int counter = 0; public static synchronized int getCount() { return counter++; } }
如果方法不是靜態方法,那麼使用關鍵字synchronized同步的將是實例對象,而不是類對象。
在這個特殊的計數例子裡,通過使用java.util.concurrent.atomic包下的AtomicInteger原子類,可以使count++操作變成原子操作,如下。
import java.util.concurrent.atomic.AtomicInteger; public class MyCounter { private static AtomicInteger counter = new AtomicInteger(0); public static int getCount() { return counter.getAndIncrement(); } }
在Java中本地變量是線程安全的。
每一個線程都會有一個自己的棧,兩個不同的線程是不會共享同一個棧的。
所有方法內部的本地變量將會在棧中分配空間,一旦當前線程的方法執行完畢,棧幀將馬上被移除。
譯文鏈接:http://www.programcreek.com/2014/02/how-to-make-a-method-thread-safe-in-java/