設計模式中的單例模式應該是被大家使用最廣泛的模式之一,但網上關于單例模式的詬病也不少,最集中的就是:在多線程的環境下,單例模式有可能返回不止一個的對象。那么到底為什么會出現這種情況呢?下面我們來看單例模式的兩種實現方式
方式一
public class Singleton {
//注意構造方法必須是私有的
private Singleton(){}
//在自己內部定義自己一個實例,是不是很奇怪?
//注意這是private 只供內部調用
private static Singleton instance = new Singleton();
//這里提供了一個供外部訪問本class的靜態方法,可以直接訪問
public static Singleton getInstance() {
return instance;
}
}
方式二
public class Singleton {
private static Singleton instance = null;
public static synchronized Singleton getInstance() {
//這個方法比上面有所改進,不用每次都進行生成對象,只是第一次
//使用時生成實例!
if (instance==null)
instance=new Singleton();
return instance;
}
}
}
方式二就是我們說的:滯后初始化(Lazy Initialization)。為什么會有滯后初始化這種實現方式出現呢?我們可用看到在第一種實現方式中無法向單例模式的構造方法傳遞參數,而使用滯后初始化的方式,我們可用在調用getInstance()方法的時候向方法中傳遞參數。
凡事有好處必然有壞處,滯后初始化的一個弊病就是在多線程或分布式的環境下有可能出現混亂:
“有時在某些情況下,使用Singleton并不能達到Singleton的目的,如有多個Singleton對象同時被不同的類裝入器裝載;在EJB這樣的分布式系統中使用也要注意這種情況,因為EJB是跨服務器,跨JVM的。” --摘自www.jdon.com-《GoF 23種設計模式解析》
“在多線程環境下,我們無法保證一個方法能夠持續運行到結束,其他線程的方法才開始運行。因而可能存在這樣一種情形:兩個線程幾乎同時嘗試初始化單例類。假設第一個方法發現單例為空,而第二個方法在此刻開始運行,它也會發現該單例為空。接下來,這兩個方法都將對該單例進行初始化。” --摘自《Java設計模式》
二、單例模式在多線程下的安全實現
《Java并發編程》一書建議使用屬于當前類的鎖進行同步,代碼如下:
public class Singleton {
private static Singleton instance = null;
// 注意這里的static非常重要,如果為對象變量則因為存在多份拷貝而起不到限制的作用
private static Object classLock = Singleton.class;
public static Singleton getInstance() {
//這個方法比上面有所改進,不用每次都進行生成對象,只是第一次
//使用時生成實例!
synchronized(classLock){
if (instance==null)
instance=new Singleton();
return instance;
}
}
}
}
在第一個線程開始滯后初始化的時候,如果有另一線程也準備開始初始化,這時候,第二個線程將停止執行,等待獲取對象classLock的鎖。當第二個線程獲取這個鎖并開始執行初始化的時候,它會發現該單例已不再為空(因為只存在該類的唯有實例,我們可以使用單個靜態鎖)
三、使用單例模式的另外一些注意點
·單例模式類不能實現Clonable接口,以防被克隆而產生多個實例
·單例模式類不能實現Serializable接口,以防被序列化而產生多個實例
-------------------------------------------------------------
生活就像打牌,不是要抓一手好牌,而是要盡力打好一手爛牌。