zoukankan      html  css  js  c++  java
  • 设计模式——单例模式

    单例模式定义:保证创建唯一实例。

    实现有很多方式但是有一个共同点:构造方法私有化。

    单例模式实现需要考虑的问题:

    • 保证单例
    • 延迟加载
    • 线程安全
    • 反反射、反序列化、反克隆

    下面各种实现方式就是上述问题的渐进解决。懒汉式+双重验证锁是最常用的。

    一、饿汉式

    public class SingletonV0 {
        //利用一个类只会被类加载一次创建唯一实例
        private static final SingletonV0 INSTANCE = new SingletonV0();
    
        private SingletonV0(){}
    
        public static SingletonV0 getSingletonVo(){
            return INSTANCE;
        }
    }

    二、懒汉式

    public class SingletonV1 {
    
        private static SingletonV1 INSTANCE;
    
        private SingletonV1(){}
    
        public static SingletonV1 getInstance(){
            if (INSTANCE == null){
                //延迟加载
                INSTANCE = new SingletonV1();
            }
            return INSTANCE;
        }
    }

    三、懒汉式+双重验证锁——最常用的方式

    public class SingletonV2 {
        /**
         * 注意细节:volatile修饰
         */
        private static volatile SingletonV2 INSTANCE;
    
        private SingletonV2(){}
    
        public static SingletonV2 getInstance(){
            //双重验证锁
            if (INSTANCE == null){
                synchronized (SingletonV2.class){
                    if (INSTANCE == null){
                        INSTANCE = new SingletonV2();
                    }
                }
            }
            return INSTANCE;
        }
    }

    四、懒汉式+双重验证锁+反反射+反序列化+反克隆

    public class SingletonV3 implements Cloneable, Serializable {
        /**
         * 注意细节:volatile修饰
         */
        private static volatile SingletonV3 INSTANCE;
    
        private static boolean IS_FIRST_CREATE = true;
        private SingletonV3(){
            //反反射
            if (IS_FIRST_CREATE){
                synchronized (SingletonV3.class){
                    if (IS_FIRST_CREATE){
                        IS_FIRST_CREATE = false;
                    }
                }
            }else{
                throw new UnsupportedOperationException("singleton object is already init().");
            }
        }
    
        public static SingletonV3 getInstance(){
            //双重验证锁
            if (INSTANCE == null){
                synchronized (SingletonV3.class){
                    if (INSTANCE == null){
                        INSTANCE = new SingletonV3();
                    }
                }
            }
            return INSTANCE;
        }
    
        /**
         * 反克隆
         */
        @Override
        protected Object clone() throws CloneNotSupportedException {
            throw new CloneNotSupportedException("singleton object not support clone");
        }
        /**
         * 反序列化
         */
        private void readObject(ObjectInputStream in) throws InvalidObjectException {
            throw new InvalidObjectException("can't deserialize singleton");
        }
    
        private void readObjectNoData() throws InvalidObjectException {
            throw new InvalidObjectException("can't deserialize singleton");
        }
    }

    五、内部静态类

    public class SingletonV4 {
    
        private SingletonV4(){}
    
        /**
         * 利用静态内部类的延迟加载
         */
        private static class SingletonHolder{
            private static final SingletonV4 INSTANCE = new SingletonV4();
        }
    
        public SingletonV4 getInstance(){
            return SingletonHolder.INSTANCE;
        }
    }

    六、枚举类

    用枚举类创建单例是《effective Java》中推荐使用的方法。

    枚举类的实质是编译器的一个语法糖,反编译后的代码:java枚举类语法糖

    public enum SingletonV5 {
    
        /**
         * Java的枚举类型其实是一个编译期的一个语法糖
         *  enum 编译后public final class
         *  INSTANCE  编译后 public static final SingletonV5 INSTANCE;
         *  静态块初始化+反克隆+反序列化
         */
        INSTANCE;
    
    }

    七、框架中的单例模式

    1、Spring单例Bean

    Spring中单例模式非常多,基本都是使用懒汉式+双重验证锁的方式创建单例的

    最有代表性的实现:Spring单例Bean。

    /* org.springframework.beans.factory.support.DefaultSingletonBeanRegistry#getSingleton(java.lang.String, boolean) */
        protected Object getSingleton(String beanName, boolean allowEarlyReference) {
            Object singletonObject = this.singletonObjects.get(beanName);
            if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
                synchronized (this.singletonObjects) {
                    singletonObject = this.earlySingletonObjects.get(beanName);
                    if (singletonObject == null && allowEarlyReference) {
                        ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
                        if (singletonFactory != null) {
                            singletonObject = singletonFactory.getObject();
                            this.earlySingletonObjects.put(beanName, singletonObject);
                            this.singletonFactories.remove(beanName);
                        }
                    }
                }
            }
            return singletonObject;
        }
  • 相关阅读:
    (网络流)ACM Computer Factory --POJ --3436
    (小数化分数)小数化分数2 -- HDU --1717
    (小数化分数)小数化分数2 --HDU --1717
    (网络流 模板 Dinic) Drainage Ditches --POJ --1273
    (网络流 模板 Edmonds-Karp)Drainage Ditches --POJ --1273
    (匹配)Oil Skimming -- hdu --4185
    (匹配 二维建图) Antenna Placement --POJ --3020
    (匹配)Antenna Placement --POJ --3020
    将截断字符串或二进制数据【转】
    C#中Abstract和Virtual 【转】
  • 原文地址:https://www.cnblogs.com/wqff-biubiu/p/12626781.html
Copyright © 2011-2022 走看看