單例模式是GOF設(shè)計(jì)模式中最簡(jiǎn)單的一個(gè),也是最常見(jiàn)的一個(gè),當(dāng)我們遇到某個(gè)類(lèi)在全局中只應(yīng)該存在一個(gè)實(shí)例的時(shí)候就需要使用單例,單例有很多種寫(xiě)法,不同的寫(xiě)法有不用好處和使用情景,下面做一個(gè)簡(jiǎn)要的歸納。
常見(jiàn)的幾種單例模式寫(xiě)法
- 餓漢模式
public class Singleton {
/**
* 優(yōu)點(diǎn):沒(méi)有線程安全問(wèn)題,簡(jiǎn)單
* 缺點(diǎn):提前初始化會(huì)延長(zhǎng)類(lèi)加載器加載類(lèi)的時(shí)間;如果不使用會(huì)浪費(fèi)內(nèi)存空間; 不能傳遞參數(shù)
*/
private static final Singleton instance = new Singleton();
private Singleton(){};
public static Singleton getInstance(){
return instance;
}
}
餓漢式的原理其實(shí)是基于classloder機(jī)制來(lái)避免了多線程的同步問(wèn)題 。
- 經(jīng)典模式
public class Singleton {
/**
* 優(yōu)點(diǎn):易于使用,延遲初始化,節(jié)約資源(又稱(chēng)懶漢式寫(xiě)法)
* 缺點(diǎn):當(dāng)有多個(gè)線程同時(shí)調(diào)用getInstance()會(huì)出現(xiàn)線程安全問(wèn)題
*/
private static final Singleton instance;
private Singleton(){};
public static Singleton getInstance(){
if(instance == null){
instance = new Singleton();
}
return instance;
}
}
平時(shí)開(kāi)發(fā)如果沒(méi)有線程安全的需求這種寫(xiě)法就可以滿足了
- 解決經(jīng)典模式的線程安全問(wèn)題
public class Singleton {
/**
* 優(yōu)點(diǎn):易于使用,延遲初始化,節(jié)約資源 同時(shí)保證了線程安全
* 缺點(diǎn):每次調(diào)用都會(huì)走synchronized,十分影響性能
*/
private static final Singleton instance;
private Singleton(){};
public synchronized static Singleton getInstance(){
if(instance == null){
instance = new Singleton();
}
return instance;
}
}
這種寫(xiě)法對(duì)性能影響比較大,如果調(diào)用不是很頻繁還可以接受,一般不推薦這種寫(xiě)法,下面有更好的寫(xiě)法。
- 綜合性能和線程安全模式
public class Singleton {
/**
* 優(yōu)點(diǎn):易于使用,延遲初始化,節(jié)約資源 同時(shí)保證了線程安全
* 缺點(diǎn):每次調(diào)用都會(huì)走synchronized,性能問(wèn)題
*/
private volatile static final Singleton instance;
private Singleton(){};
public static Singleton getInstance(){
if(instance == null){
synchronized(Singleton.class){
if(instance == null){
instance = new Singleton();
}
}
}
return instance;
}
}
一般來(lái)講這種寫(xiě)法是很推薦的寫(xiě)法,當(dāng)然這里的instance最好用volatile來(lái)修飾,如果我們?cè)趎ew SingleTon中做了很多事情(Singleton中有很多成員對(duì)象需要初始化賦值的時(shí)候)比如:
1.instance對(duì)象分配內(nèi)存;
2.初始化對(duì)象成員變量賦值;
3.將instance指向分配的內(nèi)存空間 (此時(shí)instance才不為null));
這些操作并非原子操作,java虛擬機(jī)編譯時(shí)可能會(huì)對(duì)指令重新排序,這樣2、3兩步?jīng)]法保證先后順序,這樣有的線程就可能就會(huì)拿到為null的instance,程序就會(huì)出問(wèn)題,利用volatile的禁止指令重排序優(yōu)化特性,就可以解決這個(gè)問(wèn)題。
- 解決餓漢模式提前初始化的寫(xiě)法
public class Singleton{
/**
* 優(yōu)點(diǎn):解決線程安全,延遲初始化(四大名著之 Effective Java推薦寫(xiě)法)
* 缺點(diǎn):好像沒(méi)有~_~
*/
private Singleton(){}
public static Singleton getInstance () {
return Holder.SINGLE_TON;
}
private static class Holder{
private static final Singleton SINGLE_TON = new Singleton();
}
}
好了以上五種就是java中常見(jiàn)的五種單例寫(xiě)法,最后兩種推薦使用,一般常見(jiàn)第四種,第五種以后要多多嘗試,畢竟谷歌大神推薦_.