成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

設(shè)計模式之單例模式

陸斌 / 2269人閱讀

摘要:單例模式關(guān)注的重點私有構(gòu)造器線程安全延遲加載序列化和反序列化安全反射攻擊安全相關(guān)設(shè)計模式單例模式和工廠模式工廠類可以設(shè)計成單例模式。

0x01.定義與類型

定義:保證一個類僅有一個實例,并提供一個全局訪問點

類型:創(chuàng)建型

UML

單例模式的基本要素

私有的構(gòu)造方法

指向自己實例的私有靜態(tài)引用

以自己實例為返回值的靜態(tài)的公有的方法

0x02.適用場景

像確保任何情況下都絕對只有一個實例

需要頻繁實例化然后銷毀的對象。

創(chuàng)建對象時耗時過多或者耗資源過多,但又經(jīng)常用到的對象。

有狀態(tài)的工具類對象。

頻繁訪問數(shù)據(jù)庫或文件的對象。

0x03.單例模式的優(yōu)缺點 1.優(yōu)點

在內(nèi)存里只有一個實例,減少了內(nèi)存開銷

可以避免對資源的多重占用

避免重復(fù)創(chuàng)建對象,提高性能

設(shè)置全局訪問點,嚴(yán)格控制訪問

2.缺點

沒有接口,擴展困難

違反開閉原則

0x04.單例模式的幾種實現(xiàn)方式 1.餓漢式

餓漢式:顧名思義,對象比較饑餓,所以一開始就創(chuàng)建好了。餓漢式也是單例模式的最簡單實現(xiàn)。

Java實現(xiàn)

/**
 * 餓漢式
 * 一開始就new好了
 */
public class HungrySingleton implements Serializable {

    /**
     * 可以直接new也可以適用靜態(tài)塊中創(chuàng)建
     * */
    private final static HungrySingleton hungrySingleton;

    static {
        hungrySingleton = new HungrySingleton1();
    }

    public static HungrySingleton getInstance() {
        return hungrySingleton;
    }
    /**
     * 私有構(gòu)造函數(shù)
     */
    private HungrySingleton() {}
}

餓漢式的單例模式,對象一開始就創(chuàng)建好了。不需要考慮線程安全問題。

餓漢式單例模式如果消耗資源比較多,而對象未被適用則會造成資源浪費。

2.懶漢式

懶漢式:說明類對象比較懶,沒有直接創(chuàng)建,而是延遲加載的,是第一次獲取對象的時候才創(chuàng)建。懶漢式的單例模式應(yīng)用較多。

a.第一個版本的Java實現(xiàn)(非線程安全)
/**
 * 懶漢式
 * 線程不安全
 */
public class LazySingleton {

    private static LazySingleton lazySingleton = null;

    //線程不安全,當(dāng)有兩個線程同時創(chuàng)建對象,會違背單例模式
    public static LazySingleton getInstance() {
        if (lazySingleton == null) {
            //會發(fā)生指令重排
            lazySingleton = new LazySingleton();
        }
        return lazySingleton;
    }
    private LazySingleton() {}
}

這個版本的懶漢式會出現(xiàn)線程安全的問題,當(dāng)兩個線程同時訪問getInstance()靜態(tài)方法時,lazySingleton還未創(chuàng)建,就會創(chuàng)建出兩個實例,違背了單例模式。

這里可以在getInstance()方法添加同步鎖synchronized解決,也可以在方法體添加類鎖,但是這樣相當(dāng)于完全鎖住了getInstance(),會出現(xiàn)性能問題。

推薦適用下面這種方式

b.雙重檢查鎖double check懶漢式(線程安全,通常適用這種方式)
/**
 * 懶漢式
 * 線程不安全
 */
public class LazyDoubleCheckSingleton {

    //volatile 禁止指令重排序
    private volatile static LazyDoubleCheckSingleton lazyDoubleCheckSingleton = null;

    /**
     * 在靜態(tài)方法中直接加synchronized相當(dāng)于鎖了類
     * @return
     */
    public static LazyDoubleCheckSingleton getInstance() {
        //同樣實鎖類, 指令重排序
        if (lazyDoubleCheckSingleton == null) {
            synchronized (LazyDoubleCheckSingleton.class) {
                if (lazyDoubleCheckSingleton == null) {
                    /**
                     * 1.分配內(nèi)存給這個對象
                     * 2.初始化對象
                     * 3.設(shè)置lazyDoubleCheckSingleton指向剛分配的內(nèi)存
                     * 2 3 順序有可能發(fā)生顛倒
                     * intra-thread semantics 不會改變單線程執(zhí)行結(jié)果,指令重排序
                     */
                    lazyDoubleCheckSingleton = new LazyDoubleCheckSingleton();
                }
            }
        }
        return lazyDoubleCheckSingleton;
    }
    private LazyDoubleCheckSingleton() {}
}

雙重檢查,只有對象為空的時候才會需要同步鎖,而第二次判斷是否為null,是對象是否已經(jīng)創(chuàng)建。

添加volatile關(guān)鍵字,防止指令重排序。

c.基于靜態(tài)內(nèi)部類的延遲加載方案

私有靜態(tài)類的延遲加載

public class StaticInnerClassSingleton {

    /**
     * 看靜態(tài)類的初始化鎖那個線程可以拿到
     */
    private static class InnerClass {
        private static StaticInnerClassSingleton staticInnerClassSingleton = new StaticInnerClassSingleton();
    }

    public static StaticInnerClassSingleton getInstance() {
        return InnerClass.staticInnerClassSingleton;
    }

    private StaticInnerClassSingleton () {
        if (InnerClass.staticInnerClassSingleton != null) {
            throw new RuntimeException("單例對象禁止反射調(diào)用");
        }
    }
}

將延遲初始化交給靜態(tài)類的初始化

3.容器單例

使用靜態(tài)容器方式來實現(xiàn)多單例類

public class ContainerSingleton {

    //靜態(tài)容器, 注意map不是線程安全的,如果為了線程安全可以使用HashTable或者ConcurrentHashMap
    private static Map singletonMap = new HashMap<>();

    public static void putInstance (String key, Object instance) {
        if (key != null && key.length() != 0) {
            if (!singletonMap.containsKey(key)) {
                singletonMap.put(key, instance);
            }
        }
    }

    public static Object getInstance (String key) {
        return singletonMap.get(key);
    }
}

容器單例如果要保證線程安全性,建議使用ConcurrentHashMap

通常使用容器單例情況是:單例對象比較多,需要統(tǒng)一維護。

4.枚舉單例模式(推薦使用)

枚舉單例是從JVM層面上做的限制

public enum EnumInstance {

    /**
     * 具體的單例實例
     */
    INSTANCE {
        protected void  printTest () {
            System.out.println("K.O print Test!");
        }
    };
    private Object data;
    protected abstract void printTest();
    public Object getData() {
        return data;
    }
    public void setData(Object data) {
        this.data = data;
    }
    public static EnumInstance getInstance() {
        return INSTANCE;
    }
}

后續(xù)會介紹到,單例模式完美防御了反射與序列化攻擊

5.ThreadLocal線程單例(并不是嚴(yán)格意義上的單例模式)

有一部分場景,要求對象的生命周期隨著線程

/**
 * 線程級單例模式
 */
public class ThreadLocalInstance {

    //靜態(tài)的ThreadLocal類保存對象
    private static final ThreadLocal threadLocal =
            ThreadLocal.withInitial(ThreadLocalInstance::new);

    private ThreadLocalInstance () {}

    public static ThreadLocalInstance getInstance () {
        return threadLocal.get();
    }
}

通過getInstance()獲取該線程的實例。

0x05.單例模式的序列化與反射攻擊 1.序列化攻擊

以前面餓漢式舉例

測試代碼

public class SerializableTest {

    public static void main(String[] args) throws IOException, ClassNotFoundException {
        //1.實例化
        HungrySingleton instance = HungrySingleton.getInstance();

        //2.寫入本地文件
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("singleton_file"));
        oos.writeObject(instance);

        //3.讀取
        File file = new File("singleton_file");
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream(file));
        HungrySingleton newInstance = (HungrySingleton) ois.readObject();

        //4.比較
        System.out.println(instance);
        System.out.println(newInstance);
        System.out.println(instance == newInstance);
    }
}

輸出結(jié)果

org.ko.singleton.hungry.HungrySingleton@135fbaa4
org.ko.singleton.hungry.HungrySingleton@568db2f2
false

解決方案:添加readResolve()方法

修改后

/**
 * 餓漢式
 * 一開始就new好了
 */
public class HungrySingleton implements Serializable {

    private final static HungrySingleton hungrySingleton;

    static {
        hungrySingleton = new HungrySingleton();
    }

    public static HungrySingleton getInstance() {
        return hungrySingleton;
    }

    /**
     * 寫完后,序列化對象會通過反射調(diào)用這個方法
     * 完全是ObjectInputStream寫死的,并沒有任何繼承關(guān)系
     * 其實每次序列化 反序列化 都已經(jīng)創(chuàng)建對象了,只是最后返回的這一個
     * @return
     */
    private Object readResolve () {
        return hungrySingleton;
    }

    private HungrySingleton() {}
}

輸出結(jié)果

org.ko.singleton.hungry.HungrySingleton@135fbaa4
org.ko.singleton.hungry.HungrySingleton@135fbaa4
true

為什么添加了readResolve()方法就可以了?

ObjectInputStream源碼中,讀取文件時寫死判斷是否有readResolve()方法,有調(diào)用這個方法,沒有則重新創(chuàng)建對象。

2.反射攻擊

通過反射攻擊,實例化對象創(chuàng)建出第二個單例對象

/**
 * 類加載時就已經(jīng)創(chuàng)建好對象
 */
public class ReflectTest {

    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class objectClass = HungrySingleton.class;

        Constructor constructor = objectClass.getDeclaredConstructor();
        constructor.setAccessible(true);

        //反射創(chuàng)建
        HungrySingleton instance = HungrySingleton.getInstance();

        //正常創(chuàng)建
        HungrySingleton newInstance = (HungrySingleton) constructor.newInstance();

        System.out.println(instance);
        System.out.println(newInstance);
        System.out.println(instance == newInstance);

        //StaticInnerClassSingleton類也是一樣的
    }
}

測試結(jié)果

org.ko.singleton.hungry.HungrySingleton@1540e19d
org.ko.singleton.hungry.HungrySingleton@677327b6
false

解決辦法:在構(gòu)造方法拋出異常

/**
 * 餓漢式
 * 一開始就new好了
 */
public class HungrySingleton implements Serializable {

    private final static HungrySingleton hungrySingleton;

    static {
        hungrySingleton = new HungrySingleton();
    }

    public static HungrySingleton getInstance() {
        return hungrySingleton;
    }

    /**
     * 寫完后,序列化對象會通過反射調(diào)用這個方法
     * 完全是ObjectInputStream寫死的,并沒有任何繼承關(guān)系
     * 其實每次序列化 反序列化 都已經(jīng)創(chuàng)建對象了,只是最后返回的這一個
     * @return
     */
    private Object readResolve () {
        return hungrySingleton;
    }

    private HungrySingleton() {
        /**
         * 對一開始就創(chuàng)建好了的類有效
         */
        if (hungrySingleton != null) {
            throw new RuntimeException("單例對象禁止反射調(diào)用");
        }
    }
}

再次測試輸出結(jié)果

Exception in thread "main" java.lang.reflect.InvocationTargetException
  at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
  at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62)
  at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
  at java.lang.reflect.Constructor.newInstance(Constructor.java:423)
  at org.ko.singleton.ReflectTest1.main(ReflectTest1.java:23)
Caused by: java.lang.RuntimeException: 單例對象禁止反射調(diào)用
  at org.ko.singleton.hungry.HungrySingleton2.(HungrySingleton2.java:36)
  ... 5 more

注意使用這種方式防止反射攻擊,餓漢式正常,懶漢式因為創(chuàng)建對象的時機不同還是會出現(xiàn)問題,這種方式只能做到盡量的防御。

3.關(guān)于枚舉單例模式防止序列化與反射

枚舉模式的實例天然具有線程安全性,防止序列化與反射的特性

驗證代碼

/**
 * 枚舉類測試
 */
public class SerializableTest {

    public static void main(String[] args) throws IOException, ClassNotFoundException {
        //測試枚舉類型
        EnumInstance instance = EnumInstance.getInstance();
        //設(shè)置對象
        instance.setData(new Object());

        //寫入文件
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("singleton_file"));
        oos.writeObject(instance);

        //讀取文件
        File file = new File("singleton_file");
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream(file));
        EnumInstance newInstance = (EnumInstance) ois.readObject();

        //比較實例
        System.out.println(instance);
        System.out.println(newInstance);
        System.out.println(instance == newInstance);

        //比較實例中引用對象
        System.out.println(instance.getData());
        System.out.println(newInstance.getData());
        System.out.println(instance.getData() == newInstance.getData());
    }
}

測試結(jié)果:

INSTANCE
INSTANCE
true
java.lang.Object@5fd0d5ae
java.lang.Object@5fd0d5ae
true

反射攻擊測試

/**
 * 類加載時就已經(jīng)創(chuàng)建好對象
 */
public class ReflectTest {
    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class objectClass = EnumInstance.class;

        Constructor constructor = objectClass.getDeclaredConstructor(String.class, int.class);
        constructor.setAccessible(true);

        //反射對象
        EnumInstance newInstance = (EnumInstance) constructor.newInstance("K.O", 1);
        //實例對象
        EnumInstance instance = EnumInstance.getInstance();

        System.out.println(instance);
        System.out.println(newInstance);
        System.out.println(instance == newInstance);
    }
}

測試結(jié)果,枚舉類沒辦法通過構(gòu)造函數(shù)創(chuàng)建實例

Exception in thread "main" java.lang.IllegalArgumentException: Cannot reflectively create enum objects
  at java.lang.reflect.Constructor.newInstance(Constructor.java:417)
  at org.ko.singleton.ReflectTest3.main(ReflectTest3.java:21)

枚舉類反編譯結(jié)果

//final的
public final class EnumInstance extends Enum{
    public static EnumInstance[] values(){
        return (EnumInstance[])$VALUES.clone();
    }

    public static EnumInstance valueOf(String name){
        return (EnumInstance)Enum.valueOf(org/ko/singleton/byenum/EnumInstance, name);
    }

    //私有構(gòu)造器
    private EnumInstance(String s, int i){
        super(s, i);
    }

    public Object getData(){
        return data;
    }

    public void setData(Object data){
        this.data = data;
    }

    public static EnumInstance getInstance(){
        return INSTANCE;
    }

    //static final
    public static final EnumInstance INSTANCE;
    private Object data;
    private static final EnumInstance $VALUES[];

    //通過靜態(tài)塊加載它,比較像餓漢模式
    static {
        INSTANCE = new EnumInstance("INSTANCE", 0);
        $VALUES = (new EnumInstance[] {
            INSTANCE
        });
    }
}

結(jié)論:如果不是特別重的對象,建議使用枚舉單例模式,它是JVM天然的單例。

0x06.單例模式關(guān)注的重點

私有構(gòu)造器

線程安全

延遲加載

序列化和反序列化安全

反射攻擊安全

0x07.相關(guān)設(shè)計模式

單例模式和工廠模式:工廠類可以設(shè)計成單例模式。

單例模式和享元模式:可以通過享元模式來獲取單例對象

0x08.相關(guān)代碼

單例模式:https://github.com/sigmako/design-pattern/tree/master/singleton

0x09.參考文章

慕課網(wǎng)設(shè)計模式精講: https://coding.imooc.com/class/270.html

23種設(shè)計模式(1):單例模式: https://blog.csdn.net/zhengzhb/article/details/7331369

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/74875.html

相關(guān)文章

  • 大話PHP設(shè)計模式單例模式升級版

    摘要:用來指向已創(chuàng)建好的實例構(gòu)造函數(shù)為空注意這里是關(guān)鍵這是我們需要調(diào)用的方法把函數(shù)也定義為空,這樣就大功告成啦。 接上一篇大話PHP設(shè)計模式之單例模式 這一篇介紹一下升級版的單例模式,廢話不說先上代碼 不完美的單例模式 class singleMode { //用來指向已創(chuàng)建好的實例 public static $instance; //判斷是...

    darcrand 評論0 收藏0
  • 每天一個設(shè)計模式單例模式

    摘要:博主按每天一個設(shè)計模式旨在初步領(lǐng)會設(shè)計模式的精髓,目前采用靠這吃飯和純粹喜歡兩種語言實現(xiàn)。單例模式用途如果一個類負(fù)責(zé)連接數(shù)據(jù)庫的線程池日志記錄邏輯等等,此時需要單例模式來保證對象不被重復(fù)創(chuàng)建,以達到降低開銷的目的。 博主按:《每天一個設(shè)計模式》旨在初步領(lǐng)會設(shè)計模式的精髓,目前采用javascript(_靠這吃飯_)和python(_純粹喜歡_)兩種語言實現(xiàn)。誠然,每種設(shè)計模式都有多種實...

    yy736044583 評論0 收藏0
  • 每天一個設(shè)計模式單例模式

    摘要:博主按每天一個設(shè)計模式旨在初步領(lǐng)會設(shè)計模式的精髓,目前采用靠這吃飯和純粹喜歡兩種語言實現(xiàn)。單例模式用途如果一個類負(fù)責(zé)連接數(shù)據(jù)庫的線程池日志記錄邏輯等等,此時需要單例模式來保證對象不被重復(fù)創(chuàng)建,以達到降低開銷的目的。 博主按:《每天一個設(shè)計模式》旨在初步領(lǐng)會設(shè)計模式的精髓,目前采用javascript(_靠這吃飯_)和python(_純粹喜歡_)兩種語言實現(xiàn)。誠然,每種設(shè)計模式都有多種實...

    lijy91 評論0 收藏0
  • 大話PHP設(shè)計模式單例模式

    摘要:上面是簡單的單例模式,自己寫程序的話夠用了,如果想繼續(xù)延伸,請傳送至大話設(shè)計模式之單例模式升級版 看了那么多單例的介紹,都是上來就說怎么做,也沒見說為什么這么做的。那小的就來說說為什么會有單例這個模式以便更好的幫助初學(xué)者真正的理解這個設(shè)計模式,如果你是大神,也不妨看完指正一下O(∩_∩)O首先我不得不吐槽一下這個模式名字單例,初學(xué)者通過字面很難理解什么是單例,我覺得應(yīng)該叫唯一模式更貼切...

    VEIGHTZ 評論0 收藏0
  • 優(yōu)才公開課筆記:php設(shè)計模式(一) 單例模式

    摘要:最近開展了三次設(shè)計模式的公開課,現(xiàn)在來總結(jié)一下設(shè)計模式在中的應(yīng)用,這是第一篇創(chuàng)建型模式之單例模式。不過因為不支持多線程所以不需要考慮這個問題了。 最近開展了三次設(shè)計模式的公開課,現(xiàn)在來總結(jié)一下設(shè)計模式在PHP中的應(yīng)用,這是第一篇創(chuàng)建型模式之單例模式。 一、設(shè)計模式簡介 首先我們來認(rèn)識一下什么是設(shè)計模式: 設(shè)計模式是一套被反復(fù)使用、容易被他人理解的、可靠的代碼設(shè)計經(jīng)驗的總結(jié)。 設(shè)計模式不...

    guyan0319 評論0 收藏0
  • JavaScript設(shè)計模式單例模式

    摘要:原文博客地址單例模式系統(tǒng)中被唯一使用,一個類只有一個實例。中的單例模式利用閉包實現(xiàn)了私有變量兩者是否相等弱類型,沒有私有方法,使用者還是可以直接一個,也會有方法分割線不是單例最簡單的單例模式,就是對象。 原文博客地址:https://finget.github.io/2018/11/06/single/ 單例模式 系統(tǒng)中被唯一使用,一個類只有一個實例。實現(xiàn)方法一般是先判斷實例是否存在,...

    lk20150415 評論0 收藏0

發(fā)表評論

0條評論

陸斌

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<