1、懶漢式(線程并不安全)
public class Singleton {
private static Singleton instance;
private Singleton(){}
public static Singleton getInstance(){
if(instance==null)
instance=new Singleton();
return instance;
}
}
2曹体、懶漢式(線程安全)
由于是對(duì)方法進(jìn)行synchronized 故效率十分低下
public class Singleton {
private static Singleton instance;
private Singleton(){}
public static synchronized Singleton getInstance(){
if(instance==null)
instance=new Singleton();
return instance;
}
}
3洒闸、餓漢式(線程安全)
把初始化操作交給類加載器池摧,但沒有延遲加載的效果俊嗽,效果一般
public class Singleton {
private static Singleton instance=new Singleton();
private Singleton(){}
public static synchronized Singleton getInstance(){
return instance;
}
}
4堕仔、雙重校驗(yàn)鎖(線程不安全)
該方式其實(shí)是對(duì)懶漢式的一個(gè)改進(jìn)躲因,使其延遲初始化,減少同步的開銷
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;
}
}
該方式不是線程安全的原因是:不同步情況下趁耗,對(duì)引用類型是不安全的沉唠。
舉例:若有線程A進(jìn)入synchronized中 執(zhí)行構(gòu)造函數(shù),但構(gòu)造函數(shù)未執(zhí)行完畢
線程B則在第一個(gè)instance 時(shí)可見苛败,判斷是instance則不為null 反回了部分
初始化的instance满葛,從而導(dǎo)致校驗(yàn)?zāi)J降氖?br>
使用volatile 保證順序的一致性(線程安全)
public class Singleton {
private volatile static Singleton instance;
private Singleton(){}
public static Singleton getInstance(){
if(instance==null)
synchronized(Singleton.class){
if(instance==null)
instance=new Singleton();
}
return instance;
}
}
推薦方式 IODH :Initialization on Demand Holder
(線程安全且效率高,性能好)
public class Singleton {
private Singleton(){}
static class SingletonHolder{
static final Singleton INSTANCE=new Singleton();
}
public static Singleton getInstance(){
return SingletonHolder.INSTANCE;
}
}
文檔解釋:
https://en.wikipedia.org/wiki/Initialization-on-demand_holder_idiom
大概意思是:
類是由JVM加載的著拭,類的初始化纱扭。
類沒有任何靜態(tài)變量進(jìn)行初始化,所以靜態(tài)類定義SingletonHolder內(nèi)也
沒有被初始化儡遮。直到JVM確認(rèn)SingletonHolder必須被執(zhí)行乳蛾,并且是第一次
發(fā)生時(shí)才會(huì)被初始化,又是因?yàn)閮?nèi)部類鄙币,導(dǎo)致靜態(tài)變量INSTANCE是被外層
類的構(gòu)造函數(shù)進(jìn)行初始化肃叶,該初始化階段是由JLS保證串行,即非并發(fā)十嘿,因此在加載和初始化期間因惭,靜態(tài)getInstance方法中不需要進(jìn)一步的同步。并且由于初始化階段在串行操作中寫入靜態(tài)變量INSTANCE绩衷。
雖然實(shí)現(xiàn)是一個(gè)高效的線程安全的“單例”緩存蹦魔,沒有同步開銷,并且性能比無限制的同步性能好咳燕,只有當(dāng)Something的構(gòu)造可以保證不失敗
時(shí)勿决,成語才能使用。在大多數(shù)JVM實(shí)現(xiàn)中招盲,如果Singleton的
構(gòu)造失敗低缩,則從同一個(gè)類加載器初始化它的后續(xù)嘗試將導(dǎo)致
NoClassDefFoundError失敗