溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

JAVA的單例模式實(shí)例分析

發(fā)布時(shí)間:2022-02-07 16:11:52 來源:億速云 閱讀:119 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要介紹“JAVA的單例模式實(shí)例分析”的相關(guān)知識(shí),小編通過實(shí)際案例向大家展示操作過程,操作方法簡(jiǎn)單快捷,實(shí)用性強(qiáng),希望這篇“JAVA的單例模式實(shí)例分析”文章能幫助大家解決問題。

一、單例模式是什么?

單例(Singleton)模式的定義:指一個(gè)類只有一個(gè)實(shí)例,且該類能自行創(chuàng)建這個(gè)實(shí)例的一種模式。

作用:單例模式是保證系統(tǒng)實(shí)例唯一性的重要手段。用于一個(gè)全局類對(duì)象在多個(gè)地方被使用的場(chǎng)景下,保障了整個(gè)系統(tǒng)只有一個(gè)對(duì)象被使用,很好的節(jié)約了資源

實(shí)現(xiàn)方法:將類的實(shí)例化方法私有化來防止程序通過其他方式創(chuàng)建該類的實(shí)例,提供一個(gè)全局唯一獲取該類實(shí)例的方法幫助用戶獲取類的實(shí)例。

實(shí)現(xiàn)單例模式很簡(jiǎn)單,每次獲取前先判斷系統(tǒng)是否已經(jīng)存在單例對(duì)象,沒有就創(chuàng)建,有就返回這個(gè)對(duì)象。單例模式常見的寫法有懶漢式單例和餓漢式單例。

二、懶漢式單例

定義:類加載時(shí)沒有生成單例,第一次調(diào)用 getlnstance 方法時(shí)創(chuàng)建這個(gè)單例。

public class LazySingleton {

    //定義一個(gè)私有的靜態(tài)對(duì)象instance,靜態(tài)方法和屬性屬于類,能保障單例對(duì)象的唯一性
    private static LazySingleton instance;

    //私有化構(gòu)造方法,只能在本類中被訪問,其他類中不能通過構(gòu)造方法直接創(chuàng)建對(duì)象
    private LazySingleton() {
    }

    //提供一個(gè)全局唯一獲取實(shí)例的方法
    public static synchronized LazySingleton getInstance(){
        if(instance==null){
            instance=new LazySingleton();
        }
        return instance;
    }
}

懶漢模式在獲取對(duì)象實(shí)例時(shí)做了加鎖操作,因此是線程安全的

測(cè)試代碼與結(jié)果

public class TestLazySingleton {
    public static void main(String[] args) {
        LazySingleton lazySingleton1=LazySingleton.getInstance();
        LazySingleton lazySingleton2=LazySingleton.getInstance();
        LazySingleton lazySingleton3=LazySingleton.getInstance();
        System.out.println(lazySingleton1);
        System.out.println(lazySingleton2);
        System.out.println(lazySingleton3);
    }
}

JAVA的單例模式實(shí)例分析

從上圖中可以看出雖然獲取了三次實(shí)例,但每次獲取的都是同一個(gè)實(shí)例,即一個(gè)類只有一個(gè)實(shí)例。

三、餓漢式單例

定義:該模式的特點(diǎn)是類一旦加載就創(chuàng)建一個(gè)單例,在調(diào)用 getInstance 方法之前單例已經(jīng)存在。

public class HungrySingleton {
    //類加載完成后該類的實(shí)例便已經(jīng)存在
    private static HungrySingleton instance=new HungrySingleton();
    //私有化構(gòu)造方法
    private HungrySingleton(){
    }
    //類加載后實(shí)例就存在,不會(huì)出現(xiàn)線程安全問題,不需要加鎖
    public static HungrySingleton getInstance(){
        return  instance;
    }
}

測(cè)試代碼和結(jié)果

public class TestHungrySingleton {
    public static void main(String[] args) {
       HungrySingleton hungrySingleton1=HungrySingleton.getInstance();
       HungrySingleton hungrySingleton2=HungrySingleton.getInstance();
       HungrySingleton hungrySingleton3=HungrySingleton.getInstance();
        System.out.println(hungrySingleton1);
        System.out.println(hungrySingleton2);
        System.out.println(hungrySingleton3);
    }
}

JAVA的單例模式實(shí)例分析

從上圖看出餓漢式單例在整個(gè)運(yùn)行過程中也只存在一個(gè)實(shí)例。

懶漢式單例和餓漢式單例的區(qū)別
1.懶漢模式在類中定義了單例但是并未實(shí)例化,實(shí)例化是在方法中實(shí)現(xiàn)的,而餓漢模式定義的時(shí)候就進(jìn)行了實(shí)例化
2.懶漢模式需要在獲取實(shí)例的方法上加鎖保證線程安全,餓漢模式不需要加鎖。

四、雙重校驗(yàn)鎖

懶漢模式用到了synchronized,會(huì)導(dǎo)致很大的性能開銷,并且加鎖其實(shí)只需要在第一次初始化的時(shí)候用到,之后的調(diào)用都沒必要再進(jìn)行加鎖。

雙重校驗(yàn)鎖在懶漢模式的基礎(chǔ)上做了進(jìn)一步的優(yōu)化,給靜態(tài)對(duì)象加上volatile來保證有序性,第一次獲取對(duì)象時(shí)通過synchronize(Singleton.class)保障操作的唯一性。

public class LockSingleton {
    private  volatile static LockSingleton lockSingleton;
    private LockSingleton(){}
    public static LockSingleton getInstance(){
        if(lockSingleton==null){
            synchronized (LockSingleton.class){
                if (lockSingleton==null){
                    lockSingleton=new LockSingleton();
                }
            }
        }
        return lockSingleton;
    }
}

測(cè)試代碼與結(jié)果

public class LockTest {
    public static void main(String[] args) {
        LockSingleton lockSingleton1=LockSingleton.getInstance();
        LockSingleton lockSingleton2=LockSingleton.getInstance();
        LockSingleton lockSingleton3=LockSingleton.getInstance();
        System.out.println(lockSingleton1);
        System.out.println(lockSingleton2);
        System.out.println(lockSingleton3);
    }
}

JAVA的單例模式實(shí)例分析

執(zhí)行雙重檢查是因?yàn)?,如果多個(gè)線程同時(shí)了通過了第一次檢查,并且其中一個(gè)線程首先通過了第二次檢查并實(shí)例化了對(duì)象,那么剩余通過了第一次檢查的線程就不會(huì)再去實(shí)例化對(duì)象。

除了第一次創(chuàng)建實(shí)例的時(shí)候會(huì)出現(xiàn)加鎖的情況,后續(xù)的所有調(diào)用都會(huì)避免加鎖而直接返回,解決了性能消耗的問題。

關(guān)于“JAVA的單例模式實(shí)例分析”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí),可以關(guān)注億速云行業(yè)資訊頻道,小編每天都會(huì)為大家更新不同的知識(shí)點(diǎn)。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI