美文网首页
单例模式

单例模式

作者: 石器时代小古董 | 来源:发表于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; 
  }
  
}

相关文章

  • 【设计模式】单例模式

    单例模式 常用单例模式: 懒汉单例模式: 静态内部类单例模式: Android Application 中使用单例模式:

  • Android设计模式总结

    单例模式:饿汉单例模式://饿汉单例模式 懒汉单例模式: Double CheckLock(DCL)实现单例 Bu...

  • 2018-04-08php实战设计模式

    一、单例模式 单例模式是最经典的设计模式之一,到底什么是单例?单例模式适用场景是什么?单例模式如何设计?php中单...

  • 设计模式之单例模式详解

    设计模式之单例模式详解 单例模式写法大全,也许有你不知道的写法 导航 引言 什么是单例? 单例模式作用 单例模式的...

  • Telegram开源项目之单例模式

    NotificationCenter的单例模式 NotificationCenter的单例模式分析 这种单例模式是...

  • 单例模式Java篇

    单例设计模式- 饿汉式 单例设计模式 - 懒汉式 单例设计模式 - 懒汉式 - 多线程并发 单例设计模式 - 懒汉...

  • IOS单例模式的底层原理

    单例介绍 本文源码下载地址 1.什么是单例 说到单例首先要提到单例模式,因为单例模式是单例存在的目的 单例模式是一...

  • 单例

    iOS单例模式iOS之单例模式初探iOS单例详解

  • 单例模式

    单例模式1 单例模式2

  • java的单例模式

    饿汉单例模式 懒汉单例模式

网友评论

      本文标题:单例模式

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