1 概述
單例模式有幾個好處:
(1)某些類別創建比較頻繁,對於一些大型的對象,這是一筆很大的系統開銷。
(2)省去了new操作符,降低了系統記憶體的使用頻率,並減輕GC壓力。
(3)有些類如交易所的核心交易引擎,控制交易流程,如果該類可以創建多個的話,系統完全亂了。
2 詳解
單例模式常用的寫法有如下這麼兩種。
2.1 餓漢式
如果應用程式總是創建並使用單例模式,或者在創建和運行時壓力不是很大的情況下,可以使用私有靜態變量,提前把物件創建好。
複製代碼代碼如下:
package org.scott.singleton;
/**
* @author Scott
* @version 2013-11-16
* @description
*/
public class Singleton1 {
private static Singleton1 uniqueInstance = new Singleton1();
private Singleton1(){
}
public static Singleton1 getInstance(){
return uniqueInstance;
}
}
這樣做的話,當JVM載入這個類別的時候,根據初始化的順序,就已經把物件創建好了。同時,JVM可以保證任何執行緒在存取這個單例物件之前,一定先建立此實例,並且只建立一次。
當然,也可以使用一個靜態內部類別來完成同樣的功能。
複製代碼代碼如下:
package org.scott.singleton;
/**
* @author Scott
* @version 2013-11-16
* @description
*/
public class Singleton2 {
private Singleton2() {
}
/**
* 此處使用一個內部類別來維護單例
* */
private static class SingletonFactory {
private static Singleton2 instance = new Singleton2();
}
public static Singleton2 getInstance() {
return SingletonFactory.instance;
}
/**
* 如果該物件被用於序列化,可以確保物件在序列化前後保持一致
* */
public Object readResolve() {
return getInstance();
}
}
2.2 雙重鎖方式
“雙重鎖”,顧名思義就是兩把鎖,第一把鎖用來檢查要創建的實例物件是否已經創建了,如果尚未創建才使用第二把鎖來進行同步。
複製代碼代碼如下:
package org.scott.singleton;
/**
* @author Scott
* @version 2013-11-16
* @description
*/
public class Singleton3 {
private volatile static Singleton3 uniqueInstance;
private Singleton3(){
}
public static Singleton3 getInstance(){
if(uniqueInstance == null){
synchronized(Singleton3.class){
if(uniqueInstance == null){
uniqueInstance = new Singleton3();
}
}
}
return uniqueInstance;
}
}
如果對效能要求比較高的話,這種方式可以大大減少創建的時間,目前來說,這種方式也是比較通用的一種創建單例的方式。