單例模式

單例模式的介紹

單例模式是應(yīng)用中最廣的模式之一,在應(yīng)用這個(gè)模式時(shí),單例對(duì)象的類必須保證只有一個(gè)實(shí)例的存在,一般在很消耗資源,不能夠自由構(gòu)造對(duì)象的情況下使用這種模式。

單例模式的定義

確保某一個(gè)類只有一個(gè)實(shí)例,并且這個(gè)類自己實(shí)例并向整個(gè)系統(tǒng)提供這個(gè)實(shí)例。簡(jiǎn)而言之就是,在整個(gè)應(yīng)用中只存在這個(gè)類的一個(gè)實(shí)例,而且這個(gè)類的對(duì)象就在這個(gè)類中進(jìn)行new,其他類可以調(diào)用到這個(gè)實(shí)例。

單例模式的使用場(chǎng)景

首先是確保某個(gè)類有且只有一個(gè)對(duì)象的場(chǎng)景,避免產(chǎn)生多個(gè)對(duì)象消耗過(guò)多的資源,比如:訪問(wèn)IO和數(shù)據(jù)庫(kù)、下載數(shù)據(jù)等,或者某種類型的對(duì)象只能存在一個(gè)的情況下。

實(shí)現(xiàn)單例的關(guān)鍵點(diǎn):

  1. 構(gòu)造函數(shù)不對(duì)外開(kāi)放,一般使用private
  2. 通過(guò)一個(gè)靜態(tài)方法或者枚舉返回單例類對(duì)象
  3. 確保單例類的對(duì)象有且只有一個(gè),尤其在多線程的情況下
  4. 確保單例類的對(duì)象在反序列時(shí)不會(huì)重新構(gòu)建對(duì)象

單例模式的簡(jiǎn)單示例

一個(gè)公司有多名員工,但只有一個(gè)CEO類,這里我們使用單例模式創(chuàng)建一個(gè)CEO類,CEO也是員工,所以實(shí)現(xiàn)CEO繼承員工類(員工類不寫出來(lái)了),這個(gè)模式的實(shí)現(xiàn)核心在于將CEO類的構(gòu)造方法私有化,用外部不能通過(guò)構(gòu)造函數(shù)來(lái)實(shí)例化CEO對(duì)象,而CEO類可以通過(guò)一個(gè)公有靜態(tài)方法返回一個(gè)靜態(tài)對(duì)象。

餓漢模式

餓漢單例模式是在聲明靜態(tài)對(duì)象的時(shí)候就初始化

public class CEO extends Staff{
    private static final CEO mCeo=new CEO();        //聲明對(duì)象時(shí)就實(shí)例化

    //構(gòu)造方法
    public CEO(){};

    //公有的靜態(tài)方法,對(duì)外暴露出CEO對(duì)象
    public static CEO getCEO(){
        return mCeo;
    }
}

懶漢模式

懶漢模式與餓漢模式不同的是,懶漢模式是聲明一個(gè)對(duì)象,并且在用戶第一次調(diào)用getInstant時(shí)的時(shí)候進(jìn)行初始化。

    public class Singleton{
        private static Singleton instance;

        //構(gòu)造方法私有化
        private Singleton(){};

        //使用同步方法,在多線程的情況下保證了單例對(duì)象的唯一性,但當(dāng)instance實(shí)例化后仍然會(huì)調(diào)用同步方法,這樣會(huì)消耗不必要的資源
        public static synchronized Singleton getInstance(){
            if(instance==null){
                instance=new Singleton();
            }
            return instance;
        }
    }

懶漢單例模式的優(yōu)點(diǎn): 單例只有在使用的時(shí)候才會(huì)被實(shí)例化,在一定的程度上節(jié)約了資源。
懶漢單例模式的缺點(diǎn): 第一次加載時(shí)需要及時(shí)進(jìn)行實(shí)例化,反應(yīng)稍慢,每次調(diào)用getInstance方法都要進(jìn)行同步, 造成不必要的同步開(kāi)銷。

DCL實(shí)現(xiàn)單例

DCL模式是使用最多的單例實(shí)現(xiàn)方式。

優(yōu)點(diǎn): 既能在單例使用時(shí)才初始化單例,又能保證線程安全,且單例對(duì)象初始化后調(diào)用getInstance不進(jìn)行同步鎖。

public class Singleton{
        private static Singleton instance;

        //構(gòu)造方法私有化
        private Singleton(){};

        public static Singleton getInstance(){
            if(instance==null){
                Synchronized(Singleton.class){
                    if(instance==null){
                        instance=new Singleton();
                    }
                }
            }
            return instance;
        }
    }

在getInstance方法中對(duì)instance進(jìn)行了兩次判空,第一次為了避免不必要的同步,第二次是為了在null的情況下創(chuàng)建實(shí)例。這是什么意思了?首先來(lái)看“instance=new Singleton()”這條語(yǔ)句,執(zhí)行這句代碼后計(jì)算機(jī)會(huì)大概做3件事:

  1. 給Singleton的實(shí)例分配內(nèi)存
  2. 調(diào)用Singleton()的構(gòu)造函數(shù),初始化成員字段
  3. 將instance對(duì)象指向分配的內(nèi)存空間(此時(shí)instance就不是null了)

由于java編譯器允許處理器亂序執(zhí)行,因此上面的2、3步驟的順序是無(wú)法保證,如果在3執(zhí)行完后、2未執(zhí)行前從線程A切換到線程B上,這時(shí)instance在A中執(zhí)行了第三步,instance已經(jīng)是非空的了,所以B取走instance再使用就會(huì)出錯(cuò),這是DCL失效的問(wèn)題。

DCL優(yōu)點(diǎn): 資源利用率高,第一次執(zhí)行g(shù)etInstance時(shí)單例對(duì)象才會(huì)被實(shí)例化,效率高
DLC缺點(diǎn): 第一次加載時(shí)反應(yīng)較慢,在高并發(fā)的情況下還是會(huì)有一定的缺點(diǎn),但發(fā)生的概率較低。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

友情鏈接更多精彩內(nèi)容