单例模式

参考文章

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; 
  }
  
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 单例模式(SingletonPattern)一般被认为是最简单、最易理解的设计模式,也因为它的简洁易懂,是项目中最...
    成热了阅读 4,298评论 4 34
  • 前言 本文主要参考 那些年,我们一起写过的“单例模式”。 何为单例模式? 顾名思义,单例模式就是保证一个类仅有一个...
    tandeneck阅读 2,540评论 1 8
  • 单例模式的实现 单例模式的实现一般来说有2种方式:懒汉式(延迟加载)、饿汉式(非延迟加载)。 1. 饿汉式(非延迟...
    JerryL_阅读 986评论 0 3
  • 1.单例模式概述 (1)引言 单例模式是应用最广的模式之一,也是23种设计模式中最基本的一个。本文旨在总结通过Ja...
    曹丰斌阅读 3,006评论 6 47
  • 从踏入职场的第一天起,就会面对形形色色的人,面对各种各样的问题甚至是困难。有人可能会说职场哪有你说的那么可怕,但...
    海比阅读 531评论 0 0