設計模式(一):單例模式詳解以及幾種實現方式及其優缺點

概念

單例:保證一個類僅有一個實例,並提供一個訪問它的全局訪問點。

單例模式是一種常用的軟件設計模式之一,其目的是保證整個應用中只存在類的唯一個實例。

比如我們在系統啓動時,需要加載一些公共的配置信息,對整個應用程序的整個生命週期中都可見且唯一,這時需要設計成單例模式。如:spring容器,session工廠,緩存,數據庫連接池等等。

如圖

在這裏插入圖片描述

特點

1)單例模式只能有一個實例。

2)單例類必須創建自己的唯一實例。

3)單例類必須向其他對象提供這一實例。

如何保證實例的唯一

1)保證實例化一次

2)由類本身進行實例化

3)防止外部初始化

4)對外提供獲取實例的方法

5)線程安全

幾種單例模式以及優缺點

懶漢模式

在用的時候才加載,延遲加載

線程不安全示例

public class SlackerSingletonDemo {
    private  static SlackerSingletonDemo instance;
    private SlackerSingletonDemo(){

    }
    public static SlackerSingletonDemo getInstance() {
        if (instance == null) {
            instance = new SlackerSingletonDemo();
        }
        return instance;
    }
}

優點

在獲取實例的方法中,進行實例的初始化,節省系統資源

缺點

1.如果獲取實例時,初始化工作較多,加載速度會變慢,影響系統系能

2.每次獲取實例都要進行非空檢查,系統開銷大

3.非線程安全,當多個線程同時訪問getInstance()時,可能會產生多個實例

如果需要線程安全 在getInstance 加synchronized

線程安全示例

public class SlackerSingletonDemo {
    private  static SlackerSingletonDemo instance;
    private SlackerSingletonDemo(){

}
public static synchronized SlackerSingletonDemo getInstance() {
    if (instance == null) {
        instance = new SlackerSingletonDemo();
    }
    return instance;
}

}

這種的缺點就是synchronized鎖佔用的資源浪費

餓漢模式

在定義類的靜態私有變量同時進行實例化

public class HungrySingletonDemo {
    private static HungrySingletonDemo instance = new HungrySingletonDemo();
    private HungrySingletonDemo(){

    }

    public static HungrySingletonDemo getInstance() {
        return instance;
    }
}

1.聲明靜態私有類變量,且立即實例化,保證實例化一次

2.私有構造,防止外部實例化(通過反射是可以實例化的,不考慮此種情況)

3提供public的getInstance()方法供外部獲取單例實例

好處

1.線程安全

2.獲取實例速度快

缺點

類加載即初始化實例,內存浪費

DCL模式

Double Check Lock,雙重鎖判斷機制

public class DCLSingletonDemo {
    private volatile static DCLSingletonDemo instance;
    private DCLSingletonDemo(){

    }

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

優點

線程安全,進行雙重檢查,保證只在實例未初始化前進行同步,效率高

缺點

還是實例非空判斷,耗費一定資源

靜態內部類實現模式

線程安全,調用效率高,可以延時加載

public class StaticSingletonDemo {
    private static class StaticSingletonDemoInstance{
        private static final StaticSingletonDemo instance=new StaticSingletonDemo();
    }

    private StaticSingletonDemo(){

    }
    public static StaticSingletonDemo getInstance(){
        return StaticSingletonDemoInstance.instance;
    }
}

優點:既避免了同步帶來的性能損耗,又能夠延遲加載

枚舉模式

public enum  EnumSingletonDemo {
    //枚舉元素本身就是單例
    INSTANCE;
    public void init() {
        System.out.println("資源初始化。。。");

    }
}

優點:天然線程安全,可防止反射生成實例。

如何選用

-單例對象 佔用資源少,不需要延時加載,枚舉 好於 餓漢

-單例對象 佔用資源多,需要延時加載,靜態內部類 好於 懶漢

單例模式的優缺點

優點

1.該類只存在一個實例,節省系統資源

2.對於需要頻繁創建銷燬的對象,使用單例模式可以提高系統性能。

缺點

不能外部實例化(new),調用人員不清楚調用哪個方法獲取實例時會感到迷惑,尤其當看不到源代碼時。

如果大家對java架構相關感興趣,可以關注下面公衆號,會持續更新java基礎面試題, netty, spring boot,spring cloud等系列文章,一系列乾貨隨時送達, 超神之路從此展開, BTAJ不再是夢想!

架構殿堂

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章