美文网首页
单例模式

单例模式

作者: 石器时代小古董 | 来源:发表于2018-04-09 17:07 被阅读0次

    参考文章

    https://www.jianshu.com/p/d82cbb83f393?from=timeline
    

    一、为什么要加 volatile

    在编译器中可能发生指令重排

    class T{
     int a = 10;
    }
    T t = new  T()
    在编译器中
    new #2 // 申请一个内存空间 这里是半初始化状态, a 目前等于 0 
    .....
    invoke special #2 <T.init> // 调用构造方法初始化 这里实际对变量初始化  a = 10
    ....
    arestore_1:将 t 和申请的内存空间进行关联,t 不再是空
    

    在非正常情况下,发生指令重排,会先将 t 和内存进行关联(a 还没有初始化),这时有一个线程访问 t 时,发现 t 已经指向一块内存了,直接拷贝到线程空间了,造成程序的异常。
    对线程的可见性
    每个线程都有自己的线程空间,线程会将主存中的值拷贝到自己的线程空间,很有可能拷贝的是主存还没有初始化的值,加入 volatile 关键词后,会线程空间访问这个变量时,会要求直接读取主存的值,主存发生改变时也会通知线程空间

    为防止指令重排加上volatile 关键词

    public class Singleton {
        private static (volatile) Singleton singleton;
        private Singleton(){}
        
        public static Singleton getInstance(){
            if (singleton == null){
                synchronized(Singleton.class){
                    if(singleton == null){
                        singleton = new Singleton();
                    }
                }
            }
            return singleton;
        }
    }
    

    二、使用内部类机制

    使用ClassLoader的机制,在没有使用内部类的情况下,mInstance永远不会被创建 。
    问题:可以通过反射调用到私有构造器,来创建对象

    public class SingleClass {
        private SingleClass() {
        }
    
        public static SingleClass getInstance() {
           return CreateClass.mInstance;
        }
    
        private static class CreateClass {
            private static final SingleClass mInstance = new SingleClass();
        }
    }
    
    

    三、使用DCL,保证序列化和反序列化的单例,防止反射

    public class Singleton  implements Serializable {
       //volatile关键词修饰 防止指令重排
        private static volatile Singleton singleton;
        //是否第一次调用构造器
        private static boolean isFirst=true;
        private static final long serialVersionUID=123123123254L;
     
        private Singleton(){
          if(isFirst){
             isFirst=false;
           }else{
           throw new RuntimeException("");
            }
       }
        //双锁机制
        public static Singleton getInstance(){
            // 这里的风险是当new Singleton()没有走完时
            if (singleton == null){
                synchronized(Singleton.class){
                    if(singleton == null){
                        // 1.分配内存
                        // 2.初始化对象
                        // 3.让singleton指向这个内存地址
                        // 上述的2,3两部可能因为指令重排而变为1,3,2 使用volatile防止
                        singleton = new Singleton();
                    }
                }
            }
            return singleton;
        }
       private Object readResolve(){
         return singleton; 
      }
      
    }
    

    相关文章

      网友评论

          本文标题:单例模式

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