欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java面試常見模式問(wèn)題---單例模式

 更新時(shí)間:2021年06月09日 11:49:53   作者:興趣使然の草帽路飛  
單例模式(Singleton Pattern)是 Java 中最簡(jiǎn)單的設(shè)計(jì)模式之一。這種類型的設(shè)計(jì)模式屬于創(chuàng)建型模式,它提供了一種創(chuàng)建對(duì)象的最佳方式

1、簡(jiǎn)介

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

  • 業(yè)務(wù)系統(tǒng)全局只需要⼀個(gè)對(duì)象實(shí)例,⽐如發(fā)號(hào)器、 redis 連接對(duì)象等。
  • Spring IOC容器中的 Bean 默認(rèn)就是單例。
  • Spring Boot 中的 Controller、Service、Dao 層中通過(guò) @Autowire的依賴注⼊對(duì)象默認(rèn)都是單例的。

單例模式分類

  • 懶漢:就是所謂的懶加載,延遲創(chuàng)建對(duì)象,需要用的時(shí)候再創(chuàng)建對(duì)象。
  • 餓漢:與懶漢相反,提前創(chuàng)建對(duì)象。
  • 單例模式實(shí)現(xiàn)步驟:
  • 私有化構(gòu)造函數(shù)。
  • 提供獲取單例的方法。

2、單例模式——懶漢式

單例模式——懶漢式有以下⼏種實(shí)現(xiàn)⽅式:

/**
 * @Auther: csp1999
 * @Date: 2020/11/06/20:36
 * @Description: 單例設(shè)計(jì)模式-懶漢式
 */
public class SingletonLazy {
    // 當(dāng)需要用到該實(shí)例的時(shí)候再創(chuàng)建實(shí)例對(duì)象
    private static SingletonLazy instance;
    /**
     * 構(gòu)造函數(shù)私有化
     * 不能通過(guò) new SingletonLazy() 的方式創(chuàng)建實(shí)例
     * 
     * 當(dāng)需要用到該實(shí)例的時(shí)候在加載
     * 只能通過(guò) SingletonLazy.getInstance() 這種方式獲取實(shí)例
     */
    private SingletonLazy() {
    }
    /**
     * 單例對(duì)象的方法
     */
    public void process() {
        System.out.println("方法實(shí)例化成功!");
    }
    /**
     * 方式一:
     * <p>
     * 對(duì)外暴露一個(gè)方法獲取該類的對(duì)象
     * <p>
     * 缺點(diǎn):線程不安全,多線程下存在安全問(wèn)題
     *
     * @return
     */
    public static SingletonLazy getInstance() {
        if (instance == null) {// 實(shí)例為null時(shí)候才創(chuàng)建
            /**
             * 線程安全問(wèn)題:
             * 當(dāng)某一時(shí)刻,兩個(gè)或多個(gè)線程同時(shí)判斷到instance == null成立的時(shí)候
             * 這些線程同時(shí)進(jìn)入該if判斷內(nèi)部執(zhí)行實(shí)例化
             * 則會(huì)新建出不止一個(gè)SingletonLazy實(shí)例
             */
            instance = new SingletonLazy();// 當(dāng)需要的時(shí)候再進(jìn)行實(shí)例化對(duì)象
        }
        return instance;
    }
    /**
     * 方式二:
     * 通過(guò)加synchronized鎖 保證線程安全
     *
     * 采用synchronized 對(duì)方法加鎖有很大的性能開銷
     * 因?yàn)楫?dāng)getInstance2()內(nèi)部邏輯比較復(fù)雜的時(shí)候,在高并發(fā)條件下
     * 沒(méi)獲取到加鎖方法執(zhí)行權(quán)的線程,都得等到這個(gè)方法內(nèi)的復(fù)雜邏輯執(zhí)行完后才能執(zhí)行,等待浪費(fèi)時(shí)間,效率比較低
     *
     * @return
     */
    public static synchronized SingletonLazy getInstance2() {
        if (instance == null) {// 實(shí)例為null時(shí)候才創(chuàng)建
            // 方法上加synchronized鎖后可以保證線程安全
            instance = new SingletonLazy();// 當(dāng)需要的時(shí)候再進(jìn)行實(shí)例化對(duì)象
        }
        return instance;
    }
    /**
     * 方式三:
     * 在getInstance3()方法內(nèi),針對(duì)局部需要加鎖的代碼塊加鎖,而不是給整個(gè)方法加鎖
     *
     * 也存在缺陷:
     * @return
     */
    public static SingletonLazy getInstance3() {
        if (instance == null) {// 實(shí)例為null時(shí)候才創(chuàng)建
            // 局部加鎖后可以保證線程安全,效率較高
            // 缺陷:假設(shè)線程A和線程B
            synchronized (SingletonLazy.class){
                // 當(dāng)線程A獲得鎖的執(zhí)行權(quán)的時(shí)候B等待 A執(zhí)行new SingletonLazy();實(shí)例化
                // 當(dāng)A線程執(zhí)行完畢后,B再獲得執(zhí)行權(quán),這時(shí)候還是可以實(shí)例化該對(duì)象
                instance = new SingletonLazy();// 當(dāng)需要的時(shí)候再進(jìn)行實(shí)例化對(duì)象
            }
        }
        return instance;
    }
}

單例模式:懶漢實(shí)現(xiàn) + 雙重檢查鎖定 + 內(nèi)存模型

對(duì)于上面方式三存在的缺陷,我們可以使用雙重檢查鎖定的方式對(duì)其進(jìn)行改進(jìn)

/**
 * 方式三改進(jìn)版本:
 * 在getInstance3()方法內(nèi),針對(duì)局部需要加鎖的代碼塊加鎖,而不是給整個(gè)方法加鎖
 *
 * DCL 雙重檢查鎖定 (Double-Checked-Locking) 在多線程情況下保持高性能
 *
 * 這是否安全? instance = new SingletonLazy(); 并不是原子性操作
 * jvm中 instance實(shí)例化內(nèi)存模型流程如下:
 * 1.分配空間給對(duì)象
 * 2.在空間內(nèi)創(chuàng)建對(duì)象
 * 3.將對(duì)象賦值給instance引用
 *
 * 假如出現(xiàn)如下順序錯(cuò)亂的情況:
 * 線程的執(zhí)行順序?yàn)椋? -> 3 -> 2, 那么這時(shí)候會(huì)把值寫回主內(nèi)存
 * 則,其他線程就會(huì)讀取到instance的最新值,但是這個(gè)是不完全的對(duì)象
 * (指令重排現(xiàn)象)
 *
 * @return
 */
public static SingletonLazy getInstance3plus() {
    if (instance == null) {// 實(shí)例為null時(shí)候才創(chuàng)建
        // 局部加鎖后可以保證線程安全,效率較高
        // 假設(shè)線程A和線程B 
        synchronized (SingletonLazy.class){// 第一重檢查
            // 當(dāng)線程A獲得鎖的執(zhí)行權(quán)的時(shí)候B等待 A執(zhí)行new SingletonLazy();實(shí)例化
            // 當(dāng)A線程執(zhí)行完畢后,B再獲得執(zhí)行權(quán),這時(shí)候再判斷instance == null是否成立
            // 如果不成立,B線程無(wú)法 實(shí)例化SingletonLazy
            if (instance == null){// 第二重檢查
                instance = new SingletonLazy();// 當(dāng)需要的時(shí)候再進(jìn)行實(shí)例化對(duì)象
            }
        }
    }
    return instance;
}

再次升級(jí)方式三,來(lái)解決內(nèi)存模型中的指令重排問(wèn)題

// 添加volatile 關(guān)鍵字,禁止實(shí)例化對(duì)象時(shí),內(nèi)存模型中出現(xiàn)指令重排現(xiàn)象
private static volatile SingletonLazy instance;
/**
 * 方式三再次升級(jí)版本:
 * 在getInstance3()方法內(nèi),針對(duì)局部需要加鎖的代碼塊加鎖,而不是給整個(gè)方法加鎖
 *
 * DCL 雙重檢查鎖定 (Double-Checked-Locking) 在多線程情況下保持高性能
 *
 * 解決指令重排問(wèn)題——禁止指令重排
 * @return
 */
public static SingletonLazy getInstance3plusplus() {
    if (instance == null) {// 實(shí)例為null時(shí)候才創(chuàng)建
        // 局部加鎖后可以保證線程安全,效率較高
        // 假設(shè)線程A和線程B
        synchronized (SingletonLazy.class){// 第一重檢查
            // 當(dāng)線程A獲得鎖的執(zhí)行權(quán)的時(shí)候B等待 A執(zhí)行new SingletonLazy();實(shí)例化
            // 當(dāng)A線程執(zhí)行完畢后,B再獲得執(zhí)行權(quán),這時(shí)候再判斷instance == null是否成立
            // 如果不成立,B線程無(wú)法 實(shí)例化SingletonLazy
            if (instance == null){// 第二重檢查
                instance = new SingletonLazy();// 當(dāng)需要的時(shí)候再進(jìn)行實(shí)例化對(duì)象
            }
        }
    }
    return instance;
}

單例模式——懶漢式調(diào)用:

@Test
public void testSingletonLazy(){
    SingletonLazy.getInstance().process();
}

3、單例模式——餓漢式

/**
 * @Auther: csp1999
 * @Date: 2020/11/06/21:39
 * @Description: 單例設(shè)計(jì)模式-餓漢式
 */
public class SingletonHungry {
    // 當(dāng)類加載的時(shí)候就直接實(shí)例化對(duì)象
    private static SingletonHungry instance = new SingletonHungry();
    private SingletonHungry(){}
    /**
     * 單例對(duì)象的方法
     */
    public void process() {
        System.out.println("方法實(shí)例化成功!");
    }
    public static SingletonHungry getInstance(){
        return instance;// 當(dāng)類加載的時(shí)候就直接實(shí)例化對(duì)象
    }
}

單例模式——餓漢式調(diào)用:

@Test
public void testSingletonHungry(){
    SingletonHungry.getInstance().process();
}

餓漢式單例模式,當(dāng)類加載的時(shí)候就直接實(shí)例化對(duì)象,因此不需要考慮線程安全問(wèn)題。

  • 優(yōu)點(diǎn):實(shí)現(xiàn)簡(jiǎn)單,不需要考慮線程安全問(wèn)題。
  • 缺點(diǎn):不管有沒(méi)有使用該對(duì)象實(shí)例,instance對(duì)象一直占用著這段內(nèi)存。

懶漢與餓漢式如何選擇?

  • 如果對(duì)象內(nèi)存占用不大,且創(chuàng)建不復(fù)雜,直接使用餓漢的方式即可。
  • 其他情況均采用懶漢方式(優(yōu)選)。

總結(jié)

文章會(huì)不定時(shí)更新,有時(shí)候一天多更新幾篇,如果幫助您復(fù)習(xí)鞏固了知識(shí)點(diǎn),還請(qǐng)支持一下,后續(xù)會(huì)億點(diǎn)點(diǎn)的更新!希望大家多多關(guān)注腳本之家的其他內(nèi)容!

相關(guān)文章

  • Spring-AOP 靜態(tài)普通方法名匹配切面操作

    Spring-AOP 靜態(tài)普通方法名匹配切面操作

    這篇文章主要介紹了Spring-AOP 靜態(tài)普通方法名匹配切面操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • 日歷顯示讀出輸入的年月的java代碼

    日歷顯示讀出輸入的年月的java代碼

    這篇文章主要介紹了日歷顯示讀出輸入的年月的java代碼,有需要的朋友可以參考一下
    2013-12-12
  • Java不可不知的泛型使用示例代碼

    Java不可不知的泛型使用示例代碼

    這篇文章主要介紹了Java不可不知的泛型使用,本文通過(guò)實(shí)例代碼給大家介紹了java的泛型的基本使用,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-08-08
  • Java Math.round(),Math.ceil(),Math.floor()的區(qū)別詳解

    Java Math.round(),Math.ceil(),Math.floor()的區(qū)別詳解

    這篇文章主要介紹了Java Math.round(),Math.ceil(),Math.floor()的區(qū)別詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-08-08
  • Hikari?數(shù)據(jù)庫(kù)連接池內(nèi)部源碼實(shí)現(xiàn)的小細(xì)節(jié)

    Hikari?數(shù)據(jù)庫(kù)連接池內(nèi)部源碼實(shí)現(xiàn)的小細(xì)節(jié)

    這篇文章主要介紹了Hikari?數(shù)據(jù)庫(kù)連接池內(nèi)部源碼實(shí)現(xiàn)的小細(xì)節(jié),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • 最安全的加密算法Bcrypt防止數(shù)據(jù)泄露詳解

    最安全的加密算法Bcrypt防止數(shù)據(jù)泄露詳解

    這篇文章主要為大家介紹了最安全的加密算法Bcrypt防止數(shù)據(jù)泄露詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • logback.xml動(dòng)態(tài)配置程序路徑的操作

    logback.xml動(dòng)態(tài)配置程序路徑的操作

    這篇文章主要介紹了logback.xml動(dòng)態(tài)配置程序路徑的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-02-02
  • Java多線程學(xué)習(xí)筆記

    Java多線程學(xué)習(xí)筆記

    常用的實(shí)現(xiàn)多線程的兩種方式:Thread和Runnable。之所以說(shuō)是“常用”,是因?yàn)樵贘ava 5后可以通過(guò)java.util.concurrent包中的線程池來(lái)實(shí)現(xiàn)多線程
    2021-09-09
  • Java經(jīng)典排序算法之希爾排序詳解

    Java經(jīng)典排序算法之希爾排序詳解

    這篇文章主要為大家詳細(xì)介紹了Java經(jīng)典排序算法之希爾排序,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-04-04
  • Java線程公平鎖和非公平鎖的差異講解

    Java線程公平鎖和非公平鎖的差異講解

    今天小編就為大家分享一篇關(guān)于Java線程公平鎖和非公平鎖的差異講解,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2019-01-01

最新評(píng)論