單例模式

單例模式

單例模式幾乎是最簡(jiǎn)單的模式了

public class Singleton {

    private Singleton(){};
    
    private static  Singleton singleton = null;

    public static Singleton getInstance()
    {
        if(singleton == null)
        {
            singleton = new Singleton();
        }

        return singleton;
    }
}
  • 私有化構(gòu)造方法富弦。

  • 靜態(tài)的Singleton singleton和getInstance()方法茫舶。

  • 但是上面這種做法是線程不安全的,會(huì)造成數(shù)據(jù)污染因宇。

改進(jìn)如下DCL(Double Check Lock)
public class Singleton {

    private Singleton(){};

    private static volatile Singleton singleton = null;

    public static Singleton getInstance(){

        if(singleton == null)
        {
            synchronized (Singleton.class)
            {
                if(singleton == null)
                {
                    singleton = new Singleton();
                }
            }
        }

        return singleton;
    }
}
  • volatile 關(guān)鍵字有和 synchronized 相當(dāng)?shù)男Ч婕簦情_銷更小,修飾變量彩郊。
    synchronized 修飾方法或方法塊前弯。具體參考下Java 并發(fā)編程:volatile的使用及其原理

  • getInstance()方法中需要使用同步鎖synchronized (Singleton.class)防止多線程同時(shí)進(jìn)入造成instance被多次實(shí)例化

  • 可以看到上面在synchronized (Singleton.class)外又添加了一層if,這是為了在instance已經(jīng)實(shí)例化后下次進(jìn)入不必執(zhí)行synchronized (Singleton.class)獲取對(duì)象鎖秫逝,從而提高性能恕出。

  • 總體上來(lái)說(shuō)volatile的理解還是比較困難的,如果不是特別理解违帆,也不用急浙巫,完全理解需要一個(gè)過(guò)程,在后續(xù)的文章中也還會(huì)多次看到volatile的使用場(chǎng)景刷后。這里暫且對(duì)volatile的基礎(chǔ)知識(shí)和原來(lái)有一個(gè)基本的了解的畴。總體來(lái)說(shuō)尝胆,volatile是并發(fā)編程中的一種優(yōu)化丧裁,在某些場(chǎng)景下可以代替Synchronized。但是含衔,volatile的不能完全取代Synchronized的位置煎娇,只有在一些特殊的場(chǎng)景下,才能適用volatile贪染』呵海總的來(lái)說(shuō),必須同時(shí)滿足下面兩個(gè)條件才能保證在并發(fā)環(huán)境的線程安全:

(1)對(duì)變量的寫操作不依賴于當(dāng)前值杭隙。

(2)該變量沒(méi)有包含在具有其他變量的不變式中哟绊。

說(shuō)的好復(fù)雜,說(shuō)點(diǎn)簡(jiǎn)單的寺渗,下面摘錄于《設(shè)計(jì)模式解析與實(shí)戰(zhàn)》
上面代碼這樣定義singleton ,不添加volatile關(guān)鍵字時(shí):
private static Singleton singleton = null;
假設(shè)線程A執(zhí)行到singleton = new Singleton(); 語(yǔ)句匿情,這里看起來(lái)是一句代碼兰迫,但實(shí)際上它并不是一個(gè)原子操作,這句代碼最終會(huì)編譯成多條匯編指令炬称,它大致做了3件事情汁果。
(1)給Singleton的實(shí)例分配內(nèi)存
(2)調(diào)用Singleton()的構(gòu)造函數(shù),初始化成員字段玲躯。
(3)將singleton對(duì)象指向分配的內(nèi)存空間(此時(shí)singleton就不是null了)

??但是据德,由于Java編譯器允許處理器亂序執(zhí)行,以及JDK1.5之前JMM(Java Memory Model, 即Java內(nèi)存模型)Cache跷车、寄存器到主內(nèi)存回寫順序的規(guī)定棘利,上面的第二和第三的順序是無(wú)法保證的,也就是說(shuō)朽缴,執(zhí)行順序可能是1-2-3也可能是1-3-2善玫,如果是后者,并且在3執(zhí)行完畢密强,2未執(zhí)行之前茅郎,被切換到線程B上,這時(shí)候singleton因?yàn)橐呀?jīng)在線程A內(nèi)執(zhí)行過(guò)第三點(diǎn)或渤,singleton已經(jīng)是非空了系冗,所以,線程B直接取走singleton薪鹦,再使用時(shí)就會(huì)出錯(cuò)掌敬,這就是DCL(Double Check Lock)失效問(wèn)題,而且這種難以追蹤難以重現(xiàn)的錯(cuò)誤可能會(huì)影響很久池磁。

??在JDK1.5之后奔害,SUN官方已經(jīng)注意到這種問(wèn)題,調(diào)整了JVM,具體化了volatile關(guān)鍵字框仔,因此舀武,如果JDK是1.5或之后的版本,只需要將singleton的定義改成private volatile static Singleton singleton = null; 就可以保證singleton對(duì)象每次都是從主內(nèi)存中讀取离斩,就可以使用DCL的寫法來(lái)完成單例模式银舱。當(dāng)然,volatile或多或少也會(huì)影響到性能跛梗,但考慮到程序的正確性寻馏,犧牲這點(diǎn)性能還是值得的。

內(nèi)部類單例模式

??DCL雖然在一定程度上解決了資源消耗核偿、多余的同步诚欠、線程安全等問(wèn)題,但是它還是在某些情況下出現(xiàn)了失效的問(wèn)題,這個(gè)問(wèn)題被稱為雙重檢查鎖定(DCL)失效轰绵,在《Java并發(fā)編程實(shí)踐》一書的最后談到了這個(gè)問(wèn)題粉寞,并指出這種“優(yōu)化”是丑陋的,不贊成使用左腔。而建議使用如下的代碼替換唧垦。

public class Singleton {

    private Singleton(){}
    public static Singleton getInstance(){

        return SingletonHolder.sInstence;
    }

    /**
     * 靜態(tài)內(nèi)部類
     */
    private static class SingletonHolder{

        private static final Singleton sInstence = new Singleton();
    }
}

&emsp&emsp當(dāng)?shù)谝淮渭虞dSingleton類時(shí)并不會(huì)初始化sInstence,只有在第一次調(diào)用SingletongetInstance()方法時(shí)才導(dǎo)致sInstence被初始化,因此液样,第一次調(diào)用getInstance()方法會(huì)導(dǎo)致虛擬機(jī)加載SingletonHodler類,這種方式不僅能夠確保線程安全振亮,也能夠保證單例對(duì)象的唯一性,同時(shí)也延遲了單例的實(shí)例化鞭莽,所以這是推薦使用的單例模式實(shí)現(xiàn)方式坊秸。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市澎怒,隨后出現(xiàn)的幾起案子褒搔,更是在濱河造成了極大的恐慌,老刑警劉巖喷面,帶你破解...
    沈念sama閱讀 222,946評(píng)論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件站超,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡乖酬,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,336評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門融求,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)咬像,“玉大人,你說(shuō)我怎么就攤上這事生宛∠匕海” “怎么了?”我有些...
    開封第一講書人閱讀 169,716評(píng)論 0 364
  • 文/不壞的土叔 我叫張陵陷舅,是天一觀的道長(zhǎng)倒彰。 經(jīng)常有香客問(wèn)我,道長(zhǎng)莱睁,這世上最難降的妖魔是什么待讳? 我笑而不...
    開封第一講書人閱讀 60,222評(píng)論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮仰剿,結(jié)果婚禮上创淡,老公的妹妹穿的比我還像新娘。我一直安慰自己南吮,他們只是感情好琳彩,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,223評(píng)論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般露乏。 火紅的嫁衣襯著肌膚如雪碧浊。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,807評(píng)論 1 314
  • 那天瘟仿,我揣著相機(jī)與錄音箱锐,去河邊找鬼。 笑死猾骡,一個(gè)胖子當(dāng)著我的面吹牛瑞躺,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播兴想,決...
    沈念sama閱讀 41,235評(píng)論 3 424
  • 文/蒼蘭香墨 我猛地睜開眼幢哨,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了嫂便?” 一聲冷哼從身側(cè)響起捞镰,我...
    開封第一講書人閱讀 40,189評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎毙替,沒(méi)想到半個(gè)月后岸售,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,712評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡厂画,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,775評(píng)論 3 343
  • 正文 我和宋清朗相戀三年凸丸,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片袱院。...
    茶點(diǎn)故事閱讀 40,926評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡屎慢,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出忽洛,到底是詐尸還是另有隱情腻惠,我是刑警寧澤,帶...
    沈念sama閱讀 36,580評(píng)論 5 351
  • 正文 年R本政府宣布欲虚,位于F島的核電站集灌,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏复哆。R本人自食惡果不足惜欣喧,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,259評(píng)論 3 336
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望寂恬。 院中可真熱鬧续誉,春花似錦、人聲如沸初肉。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,750評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至臼隔,卻和暖如春嘹裂,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背摔握。 一陣腳步聲響...
    開封第一講書人閱讀 33,867評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工寄狼, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人氨淌。 一個(gè)月前我還...
    沈念sama閱讀 49,368評(píng)論 3 379
  • 正文 我出身青樓泊愧,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親盛正。 傳聞我的和親對(duì)象是個(gè)殘疾皇子删咱,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,930評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容