一:?jiǎn)瘟心J降亩x
確保某個(gè)類只有一個(gè)實(shí)例。
二:線程安全
說(shuō)起單列模式,不得不提線程安全叁温,那么什么是線程安全?我們通過(guò)幾種方式來(lái)說(shuō)說(shuō)
1.多線程訪問(wèn)時(shí)核畴,采用了加鎖機(jī)制膝但,當(dāng)一個(gè)線程訪問(wèn)該類的某個(gè)數(shù)據(jù)時(shí),進(jìn)行保護(hù)谤草,其他線程不能進(jìn)行訪問(wèn)直到該線程讀取完跟束,其他線程才可使用莺奸。不會(huì)出現(xiàn)數(shù)據(jù)不一致或者數(shù)據(jù)污染。就是線程安全的泳炉。
2.你的代碼所在的進(jìn)程中有多個(gè)線程在同時(shí)運(yùn)行憾筏,而這些線程可能會(huì)同時(shí)運(yùn)行這段代碼。如果每次運(yùn)行結(jié)果和單線程運(yùn)行的結(jié)果是一樣的花鹅,而且其他的變量的值也和預(yù)期的是一樣的,就是線程安全的枫浙。
3.一個(gè)類或者程序所提供的接口對(duì)于線程來(lái)說(shuō)是原子操作刨肃,或者多個(gè)線程之間的切換不會(huì)導(dǎo)致該接口的執(zhí)行結(jié)果存在二義性,也就是說(shuō)我們不用考慮同步的問(wèn)題,就是線程安全的箩帚。
三:?jiǎn)瘟心J降膸追N寫(xiě)法
1:惡漢單列
優(yōu)點(diǎn):寫(xiě)法比較簡(jiǎn)單真友,在類裝載的時(shí)候就完成實(shí)例化。避免了線程同步問(wèn)題紧帕,天生線程安全盔然。
缺點(diǎn):類裝載的時(shí)候就完成實(shí)例化,一定程度上會(huì)造成資源的浪費(fèi)是嗜。
public class Singleton {
private static final Singleton INSTANCE=new Singleton();
private Singleton(){
}
public static Singleton getInstance(){
return INSTANCE;
}
}
2:懶漢單列
優(yōu)點(diǎn):只有使用時(shí)愈案,才被實(shí)例化。
缺點(diǎn):效率低鹅搪,第一次加載需要實(shí)例化站绪,反應(yīng)稍慢。每次調(diào)用getInstance方法都會(huì)進(jìn)行同步丽柿,消耗不必要的資源恢准。
public class Singleton {
private static Singleton instance;
private Singleton(){
}
public static synchronized Singleton getInstance(){
if(instance == null){
instance = new Singleton();
}
return instance;
}
}
3:雙重檢查單列
優(yōu)點(diǎn):資源利用率高,效率高甫题。
缺點(diǎn):第一次加載時(shí)反應(yīng)稍慢馁筐。某些情況下失效。
public class Singleton {
private static Singleton instance;
private Singleton(){
}
public static Singleton getInstance(){
if(instance == null){
synchronized (Singleton.class){
if(instance == null){
instance = new Singleton();
}
}
}
return instance;
}
}
4:靜態(tài)內(nèi)部類單列(推薦使用)
優(yōu)點(diǎn):保證了線程安全坠非,資源利用率高敏沉,效率高。
public class Singleton {
private static Singleton instance;
private Singleton() {
}
public static class SingletonInstance {
private static final Singleton INSTANCE = new Singleton();
}
public static Singleton getInstance() {
return SingletonInstance.INSTANCE;
}
}
5:枚舉單列(推薦使用)
優(yōu)點(diǎn):寫(xiě)法簡(jiǎn)單麻顶,任何情況下都是一個(gè)單列赦抖。前面幾種方法會(huì)在反序列化的時(shí)候,重新創(chuàng)建對(duì)象辅肾。
缺點(diǎn):實(shí)例化一個(gè)單例類的時(shí)候队萤,必須要記住使用相應(yīng)的獲取對(duì)象的方法。并且枚舉在SDK1.5中才添加矫钓。
public enum SingletonEnum {
INSTANCE;
public void doSomething() {
}
}
五:其他方式
除了以上方式要尔,還可以使用容器形式實(shí)現(xiàn)單列
public class SingletonManager {
private static Map<String, Object> objMap = new HashMap<>();
private SingletonManager() {
}
public static void putObject(String key, String instance){
if(!objMap.containsKey(key)){
objMap.put(key, instance);
}
}
public static Object getObject(String key){
return objMap.get(key);
}
}
四:備注
不管哪種方式實(shí)現(xiàn)舍杜,核心原理都是講構(gòu)造函數(shù)私有化,通過(guò)靜態(tài)方法獲取一個(gè)唯一的實(shí)例赵辕,且保證線程安全既绩、防止反序列化導(dǎo)致重新生成實(shí)例對(duì)象等問(wèn)題。
實(shí)現(xiàn)方式取決項(xiàng)目本身还惠。
另外饲握,附上防止反序列化時(shí)重新生成對(duì)象方法:
private Object readResolve() throws ObjectStreamException {
return instance;
}