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;
}
}
如果對性能要求比較高的話,這種方式可以大大減少創建的時間,目前來說,這種方式也是比較通用的一種創建單例的方式。