介紹
單例模式是一種創(chuàng)建型設(shè)計(jì)模式 ??,能夠保證一個(gè)類只有一個(gè)實(shí)例饲趋,并提供一個(gè)訪問(wèn)該實(shí)例的全局節(jié)點(diǎn)妓雾。
特點(diǎn)
- 保證一個(gè)類只有一個(gè)實(shí)例。最常見(jiàn)的原因是控制某些共享資源玄窝,例如數(shù)據(jù)庫(kù)或文件的訪問(wèn)權(quán)限牵寺。它的運(yùn)作方式是這樣的,如果此時(shí)你創(chuàng)建了一個(gè)對(duì)象哆料,過(guò)一會(huì)你決定再創(chuàng)建一個(gè)對(duì)象缸剪,但是此時(shí)你會(huì)獲得剛才已經(jīng)創(chuàng)建好的對(duì)象,而不會(huì)產(chǎn)生一個(gè)新對(duì)象东亦。?? 要注意的是杏节,普通構(gòu)造函數(shù)無(wú)法實(shí)現(xiàn),構(gòu)造函數(shù)的設(shè)計(jì)決定了它一定要產(chǎn)生一個(gè)新對(duì)象典阵。
- 為該實(shí)例提供一個(gè)全局訪問(wèn)節(jié)點(diǎn)奋渔,單例模式允許在程序任何地方訪問(wèn)特定對(duì)象。但是它可以保護(hù)該實(shí)例不會(huì)被其他代碼所覆蓋壮啊。
解決方案
將默認(rèn)構(gòu)造函數(shù)設(shè)置為私有嫉鲸,防止其他對(duì)象使用單例類的 new 關(guān)鍵字進(jìn)行實(shí)例化。
新建一個(gè)靜態(tài)構(gòu)造方法作為構(gòu)造函數(shù)歹啼。該函數(shù)會(huì)調(diào)用私有構(gòu)造函數(shù)來(lái)創(chuàng)建對(duì)象玄渗,無(wú)論何時(shí)調(diào)用該方法座菠,它總是會(huì)返回相同的對(duì)象。
public class Singleton {
public static Singleton instance;
private Singleton() {}
public static Singleton getInstance() {
...
...
...
}
}
懶漢模式
public static Singleton getInstance() {
if (Objects.isNull(instance)) {
instance = new Singleton();
}
return instance;
}
這種方式實(shí)現(xiàn)起來(lái)十分簡(jiǎn)單藤树,但是如果多線程訪問(wèn)單例對(duì)象浴滴,無(wú)法保證線程安全。
public static synchronized Singleton getInstance() {
if (Objects.isNull(instance)) {
instance = new Singleton();
}
return instance;
}
使用 synchronized 關(guān)鍵字控制多線程訪問(wèn)岁钓,同一時(shí)刻只有一個(gè)線程可以進(jìn)入代碼塊升略,確保了線程安全。但是這種方式會(huì)降低程序運(yùn)行效率屡限。
public static Singleton getInstance() {
if (Objects.isNull(instance)) {
synchronized (Singleton.class) {
if (Objects.isNull(instance)) {
instance = new Singleton();
}
}
}
return instance;
}
使用雙重校驗(yàn)的方式品嚣,既確保了線程安全,也不會(huì)影響程序運(yùn)行效率钧大。但是有出現(xiàn)空指針的風(fēng)險(xiǎn)翰撑。
public static Singleton getInstance() {
return StaticSingletonHolder.INSTANCE;
}
private static class StaticSingletonHolder {
private static final Singleton INSTANCE = new Singleton();
}
使用靜態(tài)內(nèi)部類的方式,確保不會(huì)重復(fù)創(chuàng)建對(duì)象啊央,也不會(huì)影響程序效率额嘿。并且還是懶加載的方式,內(nèi)存方面也處理的非常好劣挫。
餓漢模式
private static final Singleton INSTANCE = new Singleton();
public static Singleton getInstance() {
return INSTANCE;
}
使用 final 靜態(tài)成員變量册养,當(dāng)類第一次加載到內(nèi)存中的時(shí)候就完成了初始化,保證了線程安全压固。
public static SingletonUsingEnum getInstance() {
return SingletonUsingEnum.INSTANCE;
}
private enum SingletonUsingEnum {
INSTANCE;
}
使用枚舉的方式球拦,可以保證不會(huì)重復(fù)創(chuàng)建對(duì)象,線程安全帐我。還可以避免反射和序列化等帶來(lái)的問(wèn)題坎炼。是實(shí)現(xiàn)單例模式最好的方式。
總結(jié)
這是重新學(xué)習(xí)單例模式的筆記拦键,其中可能有很多地方寫的不對(duì)谣光,寫得不好,歡迎大家指正 ??芬为。