美文网首页程序员
设计模式(一)单例模式

设计模式(一)单例模式

作者: Lutils | 来源:发表于2017-03-17 21:31 被阅读78次

    要求:构造函数设置为private、通过静态方法或枚举返回单例对象
    确保单例对象有且只有一个(尤其是多线程环境)、且反序列化时不会重新构建对象

    步骤:构造函数私有化->暴露公共静态方法->确保线程安全

    饿汉模式

    
    class Singleton {
    
        // 构造函数私有化
        private Singleton() {}
        // 共有静态方法、暴露获取单例对象接口
        private static Singleton instance = new Singleton();
    
        public static Singleton getInstance() {
            return instance;
        }
    }
    
    

    懒汉模式

    优点:单例只有在使用时才会被实例化
    缺点:第一次加载时反应稍慢;每次调用getInstance都进行同步,造成不必要的同步开销

    
    class Singleton {
    
        private static Singleton instance;
        // 构造函数私有化
        private Singleton() {}
        //添加了synchronized关键字,保证多线程单例对象唯一
        public static synchronized Singleton getInstance() {
            if (instance == null) {
                instance = new Singleton();
            }
            return instance;
        }
    }
    
    

    Double Check Lock(DCL)模式

    优点:解决了懒汉模式的大部分缺点、资源利用率高
    缺点:第一次加载时反应稍慢

    
    class Singleton3 {
    
        private volatile static Singleton instance = null;
        // 构造函数私有化
        private Singleton() {}
        public static Singleton getInstance() {
            //第一次判空:避免不必要的同步
            if (instance == null) {
                synchronized (Singleton.class) {
                    //第二次判空:创建实例
                    if (instance == null) {
                        instance = new Singleton();
                    }
                }
            }
            return instance;
        }
    }
    
    

    静态内部类模式

    优点:确保线程安全、保证单例、延迟了单例的实例化

    
    class Singleton {
    
        // 构造函数私有化
        private Singleton() {}
        public static Singleton getInstance() {
            return SingletonHolder.instance;
        }
        // 静态内部类
        private static class SingletonHolder{
            private static final Singleton instance = new Singleton();
        }
    }
    
    

    以上单例方法在反序列化情况需加入以下方法

    
    private Object readResolve() throws ObjectStreamException {
        //反序列化直接返回对象,而不是生成新对象
        return instance;
    }
    
    

    因为序列化可以将一个单例的实例对象写入磁盘,然后再读出来,从而有效的获取一个实例,
    即使构造函数私有,反序列化依旧可以通过特殊途径获得新的实例

    枚举单例

    线程安全且单例,反序列化情况也不会生成新的实例

    
    public enum Singleton {
        INSTANCE;
    }
    
    

    个人博客 请戳:http://lutils.cn

    相关文章

      网友评论

        本文标题:设计模式(一)单例模式

        本文链接:https://www.haomeiwen.com/subject/juyynttx.html