閱讀663 返回首頁    go 技術社區[雲棲]


Java創建線程安全的方法

首先來看一個問題:

下麵這個方法是線程安全的嗎?如何才能讓這個方法變成線程安全的?

1 public class MyCount {
2   private static int counter = 0;
3  
4   public static int getCount(){
5     return counter++;
6   }
7 }

首先,這個方法不是線程安全的,因為counter++操作不是一個原子性的操作,也就意味著counter++操作包含了好幾個原子性的操作。實際上,counter++包含了三個原子性的操作,第一步是獲取counter的值,第二步是對counter的值加1,第三步是寫入的操作。在多線程環境對getCount()方法的調用,可能會出現下麵的場景:

thread-safe-method1

方法1:

對這個方法增加同步的控製,會讓這個方法變成線程安全的。當給靜態方法添加synchronized關鍵字修飾的時候,實際上鎖定的是這個類所對應的Class對象。在JVM中,一個類隻會存在一個Class對象。

代碼示例如下:

1 public class MyCount {
2 private static int counter = 0;
3  
4 public static synchronized int getCount(){
5 return counter++;
6 }
7 }

如果這個方法不是靜態的,那麼給方法添加synchronized關鍵字修飾的時候,鎖住的實際上是相應的實例對象,而不是這個類所對應的Class對象。

方法2:

在這個特殊的計數器的例子當中,實際上隻要把counter++操作變成原子操作,就可以讓這個方法變成是線程安全的方法。在jdk5的線程庫,java.util.concurrent.atomic包中提供的AtomicInteger類可以滿足我們的需求。

代碼示例如下:

1 public class MyCount {
2 private static AtomicInteger counter = new AtomicInteger(0);
3  
4 public static int getCount(){
5 return counter.getAndIncrement();
6 }
7 }


最後更新:2017-05-23 17:32:15

  上一篇:go  並發網係列文章集
  下一篇:go  JVM性能優化(三):垃圾收集