歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
您现在的位置: Linux教程網 >> UnixLinux >  >> Linux編程 >> Linux編程

Java中怎樣創建線程安全的方法

面試問題:

下面的方法是否線程安全?怎樣讓它成為線程安全的方法?

class MyCounter {
    private static int counter = 0;
    public static int getCount() {
        return counter++;
    }
}

本篇文章將解釋一個常見的面試題,該問題被谷歌和很多其它公司問起過。它涉及的相對比較初級,而不是關於怎樣去設計復雜的並發程序。

首先,這個問題的答案是No,因為counter++操作不是一個原子操作,而是由多個原子操作組成。

舉個例子,在如下情況:一個線程正在訪問該數據,另一個線程正在執行遞增操作;

當線程Thread 1在t1時刻訪問該方法,線程Thread 2有可能還沒執行完這個方法的操作。因此,返回線程Thread 1的值有可能還沒被遞增過。

使getCount方法成為線程安全-方式一

使用關鍵字synchronized修飾getCount方法可以使它線程安全。當使用synchronized修飾靜態方法,該類對象成為了鎖。

使用synchronized就足夠了嗎,答案是Yes.

class MyCounter {
    private static int counter = 0;
    public static synchronized int getCount() {
        return counter++;
    }
}

如果方法不是靜態方法,那麼使用關鍵字synchronized同步的將是實例對象,而不是類對象。

使getCount方法成為線程安全-方式二

在這個特殊的計數例子裡,通過使用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/

Copyright © Linux教程網 All Rights Reserved