单例模式

作者: rbty | 来源:发表于2019-07-16 20:09 被阅读0次

    单例模式

    • 单例模式可以分为饿汉式和懒汉式
    • 饿汉式会在类装载的时候变完成实例化,如果从未使用过这个实例则会造成内存浪费
    • 而懒汉式则是在需要的时候由使用者自行创建实例,这里的问题是如何在多线程环境下保证单例

    单例模式的实现方式

    饿汉式(静态常量)

    注:静态常量位于虚拟机内存的方法区,是线程共享的。

    public class Singleton {
    
        private final static Singleton INSTANCE = new Singleton();
    
        private Singleton(){}
    
        public static Singleton getInstance(){
            return INSTANCE;
        }
    }
    
    懒汉式(线程不安全)

    实现懒汉式单例,我们首先想到的是构造函数私有化以及获取唯一实例的方法

    public class Singleton {
    
        private static Singleton singleton;
    
        private Singleton() {}
    
        public static Singleton getInstance() {
            if (singleton == null) {
                singleton = new Singleton();
            }
            return singleton;
        }
    }
    
    懒汉式(方法锁,线程安全)

    于是为了保证线程安全,我们在获取实例的方法上加锁,这样的确可以保证线程安全,但是这里的同步块并非最小粒度

    public class Singleton {
    
        private static Singleton singleton;
    
        private Singleton() {}
    
        public static synchronized Singleton getInstance() {
            if (singleton == null) {
                singleton = new Singleton();
            }
            return singleton;
        }
    }
    
    懒汉式(线程不安全)

    于是我们理所当然的想到了这种错误的方式,他的问题在于若两个线程同时到达 if (singleton == null) 则会创建两个实例。

    public class Singleton {
    
        private static Singleton singleton;
    
        private Singleton() {}
    
        public static Singleton getInstance() {
            if (singleton == null) {
                synchronized (Singleton.class) {
                    singleton = new Singleton();
                }
            }
            return singleton;
        }
    }
    
    懒汉式(双重检查,仍然线程不安全)

    接下来的方案仍然是线程不安全的,但是理解他的问题我们需要引入指令重排序的概念。

    • 指令重排序 简单的来说就是,由于cpu的效率远高于内存,为了提高代码的执行速度,jvm会在不影响最终结果的前提下,对需要执行语句进行顺序调整。
    • 然后我们看代码,在执行instance = new Singleton2() 的时候,jvm大概租了三件事:1分配内存、2创建实例、3将instance指向分配的内存空间。
    • 我们无法保证执行顺序一定是123,如果被重排序为了132,在3执行完毕时,另一个线程进入同步块 判断install是否为空时,因instace已经指向了分配的内存空间,会得到false,返回一个未完成实例化的instance。
    public class Singleton2 {
    
        private Singleton2(){}
        
        private static Singleton2 instance;
    
        public static synchronized Singleton2 getInstance(){
            if(instance == null) {
                synchronized (Singleton2.class){
                    if (instance == null){
                        instance = new Singleton2();
                    }
                }
            }
            return instance;
        }
    }
    
    懒汉式(双重检查禁止重排序)

    对于多线程下的重排序问题java给出的解决方案是volatile关键字,他保证了在写操作完成前,不会对其进行读操作。

    public class Singleton2 {
    
        private Singleton2(){}
        
        private static volatile Singleton2 instance;
    
        public static synchronized Singleton2 getInstance(){
            if(instance == null) {
                synchronized (Singleton2.class){
                    if (instance == null){
                        instance = new Singleton2();
                    }
                }
            }
            return instance;
        }
    }
    
    静态内部类
    public class Singleton2 {
    
        private Singleton2(){}
        
        public static class SingletonHolder{
            private static final Singleton2 INSTANCE = new Singleton2();
        }
        
        public static Singleton2 getInstance(){
            return SingletonHolder.INSTANCE;
        }
    }
    

    相关文章

      网友评论

        本文标题:单例模式

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