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

JAVA設(shè)計模式之單例模式詳解

 更新時間:2022年01月28日 16:49:38   作者:JinziH Never Give Up  
大家好,本篇文章主要講的是JAVA設(shè)計模式之單例模式詳解,感興趣的同學(xué)趕快來看一看吧,對你有幫助的話記得收藏一下

前言

在之前的文章里已經(jīng)介紹了設(shè)計模式以及設(shè)計原則的概念,接下來我們從單例模式入手深入學(xué)習(xí)幾種常用的JAVA設(shè)計模式,在實踐中加深理解。

一、單例模式是什么?

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

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

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

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

二、懶漢式單例

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

public class LazySingleton {

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

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

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

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

測試代碼與結(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);
    }
}

在這里插入圖片描述

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

三、餓漢式單例

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

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

測試代碼和結(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);
    }
}

在這里插入圖片描述

從上圖看出餓漢式單例在整個運行過程中也只存在一個實例。

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

四、雙重校驗鎖

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

雙重校驗鎖在懶漢模式的基礎(chǔ)上做了進一步的優(yōu)化,給靜態(tài)對象加上volatile來保證有序性,第一次獲取對象時通過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;
    }
}

測試代碼與結(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);
    }
}

在這里插入圖片描述

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

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

總結(jié)

關(guān)于單例模式本文介紹了懶漢模式、餓漢模式和雙重校驗鎖。懶漢和餓漢模式的最大區(qū)別是定義實例的時候是否進行實例化。雙重校驗鎖是對懶漢模式的優(yōu)化,解決了性能消耗的問題。

到此這篇關(guān)于JAVA設(shè)計模式之單例模式詳解的文章就介紹到這了,更多相關(guān)JAVA單例模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java實現(xiàn)數(shù)據(jù)庫連接池的方法

    Java實現(xiàn)數(shù)據(jù)庫連接池的方法

    這篇文章主要介紹了Java實現(xiàn)數(shù)據(jù)庫連接池的方法,涉及java數(shù)據(jù)庫連接池的創(chuàng)建、連接、刷新、關(guān)閉及狀態(tài)獲取的常用技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-07-07
  • 教你怎么使用Java實現(xiàn)WebSocket

    教你怎么使用Java實現(xiàn)WebSocket

    這篇文章主要介紹了教你怎么使用Java WebSocket,文中有非常詳細的代碼示例,對正在學(xué)習(xí)java的小伙伴們有很好的幫助,需要的朋友可以參考下
    2021-05-05
  • 深入理解Java對象的序列化與反序列化的應(yīng)用

    深入理解Java對象的序列化與反序列化的應(yīng)用

    本篇文章是對Java中對象的序列化與反序列化進行了詳細的分析介紹,需要的朋友參考下
    2013-05-05
  • SpringMVC+MyBatis分頁(最新)

    SpringMVC+MyBatis分頁(最新)

    本篇文章主要介紹了SpringMVC+MyBatis分頁,具有一定的參考價值,有興趣的可以了解一下。
    2016-12-12
  • 淺談Spring Boot日志框架實踐

    淺談Spring Boot日志框架實踐

    這篇文章主要介紹了淺談Spring Boot日志框架實踐,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-03-03
  • springboot自定義攔截器的方法

    springboot自定義攔截器的方法

    這篇文章主要為大家詳細介紹了springboot自定義攔截器的方法,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • 關(guān)于java中@Async異步調(diào)用詳細解析附代碼

    關(guān)于java中@Async異步調(diào)用詳細解析附代碼

    本文主要介紹了java關(guān)于@Async異步調(diào)用詳細解析附代碼,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • 詳解Java高階語法Volatile

    詳解Java高階語法Volatile

    這篇文章主要介紹了Java高階語法Volatile,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03
  • java:try...catch跳過異常繼續(xù)處理循環(huán)問題

    java:try...catch跳過異常繼續(xù)處理循環(huán)問題

    這篇文章主要介紹了java:try...catch跳過異常繼續(xù)處理循環(huán)問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • Spring Boot JPA中java 8 的應(yīng)用實例

    Spring Boot JPA中java 8 的應(yīng)用實例

    這篇文章主要介紹了Spring Boot JPA中java 8 的應(yīng)用實例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-02-02

最新評論