美文网首页
单例模式

单例模式

作者: loveinthesweet | 来源:发表于2019-10-20 21:53 被阅读0次

    单例模式是简单的设计模式之一,属于创建型模式,它提供了一种创建对象的方式,确保只有单个对象被创建。这个设计模式主要目的是想在整个系统中只能出现类的实例,即一个类只有一个对象。单列模式的解决的痛点就是节约资源,节省时间从两个方面看:

    1. 由于频繁使用的对象,可以省略创建对象所花费的时间,这对于那些重量级的对象而言,是很重要的。

    2. 因为不需要频创建对象,我们的GC压力也减轻了,而在GC中会有STW(stop the world), 从这一方面也节约了GC的时间,单例模式的缺点:简单的单例模式设计开发都比较简单,但是复杂的单例模式需要考虑线程安全等并发问题。

    单例类什么时候被初始化呢? 类加载的时候就会被初始化,java虚拟机规范严格规定了有且只有四种情况必须立即对类进行初始化,遇到new、getStatic、putStatic、invokeStatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。生成这四条指令最常见的java代码场景:

    1> 使用new关键字实例化对象

    2> 读取一个类的静态字段(被final修饰、已在编译期把结果放在常量池的静态字段除外)

    3> 设置一个类的静态字段(被final修饰、已在编译期把结果放在常量池的静态字段除外)

    4> 调用一个类的静态方法

    现在我们已知道类初始化的四种情况了。

    那么我们来列举:八种单例实现方式,再分析其特点。

    1. 饿汉式静态块单例:

    public class HungryStaticSingleton{
       private static final HungryStaticSingleton instance;
       static {
          instance = new HungryStaticSingleton();
       }
       private HungryStaticSingleton(){}
       public static HungryStaticSingleton getInstance(){return instance;}

    2. 懒汉式双重检查单例:

    public class LazyDoubleCheckSingleton{
        private static volatile static LazyDoubleCheckSingleton lazy = null;
        private LazyDoubleCheckSingleton(){}
        public static LazyDoubleCheckSingleton getInstance() {
             if (null == lazy) {
                 synchronized(LazyDoubleCheckSingleton.class) {
                      if (null ==  lazy) {
                          lazy = new LazyDoubleCheckSingleton();
                      }
                    }
                }
             return lazy;
          }
    /**
    加synchronized不必多说了,保证线程同步,原子性。
    那为什么在静态常量中加volatile呢?
    1. volatile 是在内存中可见性的。
    2. 防止指令重排序:也就是说在new LazyDoubleCheckSingleton时指令重排序导致其他线程获取到未初始化完的对象。
        instance = new LazyDoubleCheckSingleton()这句,这并非是一个原子操作。事实上在JVM中这句话大概做了下面三件     事。
             1> lazy分配内存,
             2> 调用LazyDoubleCheckSingleton构造函数来初始化成员变量
             3>  将lazy对象指向分配的内存空间(执行完之后就不再为null了),但是在JVM的即时编译期中存在指令排序的优化。
                  也就是说2> 和3> 的顺序是不能保证的,最终的执行顺序可能是1>2>3也有可能是1>3>2,如果是1>3>2的话,                  则在3> 执行完毕、2> 未执行之前,被线程二抢占了,这时lazy已经是非null了(但却没有初始化),所以线程二会                直接返回lazy,但是无法使用。
      3. 内存屏障参考: https://zhuanlan.zhihu.com/p/43526907
    */

    3. 懒汉式静态内部类方式实现单列

    // 性能优于双重检查的懒汉模式
    // 使用内部类可以避免多线程环境下不安全的问题,
    // JVM对一个类的初始化会做限制,
    // 同一个时间只会允许一个线程去初始化一个类,
    // 这样从JVM层面避免了大部分单例实现的问题
    public class LazyInnerClassSingleton{
       // 默认使用LazyInnerClassSingleton,先初始化内部类
       // 如果没使用的话,内部类是不加载的
       // 为什么在私有的构造函数中加判空判断呢,是为了防止通过反射方式来创建实例
       // 强制用指定的静态方法来实例化实例
       private LazyInnerClassSingleton(){
          if (LazyHolder.LAZY != null) {
             throw new RuntimeException("不允许创建多个实例!")
          }
       }
      
      // 每一个关键字都不是多余的
      // static是为了使单例的空间共享
      // 保证这个方法不会被重写,重载
      public static LazyInnerClassSingleton getInstance() {
        // 在返回结果之前,一定会先加载内部类。
         return LazyHolder.LAZY;
      }

      // 默认不加载  
      private static class LazyHolder{
         private static final LazyInnerClassSingleton LAZY = new LazyInnerClassSingleton();
      }
    }
    /**
    静态内部类实现单例,静态内部类不被调用时,默认是不会加载的。LazyInnerClassSingleton加载时,并不需要立即加载LazyHolder,内部类不被加载则不会去初始化,故不占用内存。只有当LazyInnerClassSingleton第一次被加载时, 且调用getInstance()方法时,调用了内部类,此时内部类去加载,这种方法不仅能确保线程安全,也能保证单例的唯一性,同时也延迟了单例的实例化。无论多少个线程去创建,都只会返回一个实例,返回的地址都是同一个。
    */

    4. 枚举单例  

       public enum EnumSingleton{
          INSTANCE;
          private Object data;
          public Object getData() {return data;}
          public static EnumSingleton getInstance(){return INSTANCE;}
       }
    《Effective Java》Josh Bloch推荐使用此方法实例单例,线程安全,并发好,抵御反射攻击,序列化和反序列化安全。

    5. 容器式单例

    // Spring中bean的获取也是通过这种方式
    public class ContainerSingleton{
         private ContainerSingleton(){}
         private static Map<String, Object> ioc = new ConcurrentHashMap<>();
         public static Object getInstance(String className){
              synchronized(ioc) {
                 if (!ioc.containsKey(className)) {
                    Object obj = null;                  
                    try{
                       obj = Class.forName(className).newInstance();
                       ioc.put(className, obj);
                     }catch(Exception e) {
                         e.printStackTrance();
                     }         
                } 
               return ioc.get(className);    
            }
         }

    6. 序列化单例

        public class SeriableSingleton implements Seria{
             private static final SeriableSingleton INATANCE = new SeriableSingleton();
             private SeriableSingleton(){}
             public static SeriableSingleton getInstance(){
                 return INATANCE;
             }
             private Object readResolve(){return INATANCE;}
        }
    /**
    序列化就是把内存中的状态通过转成字节码的形式,从而转换一个IO流,写入到其他地方(可以是磁盘、网络IO)
    内存中状态给永久保存下来了。
    反序列化就是将已经持久化的字节码内容、转换为IO流,通过IO流的读取,进而将读取的内容转换为Java对象
    在转换过程中会重新创建对象new。
    为什么要加readResolve()这个方法呢? 参考:https://blog.csdn.net/u014653197/article/details/78114041
    */

    7. ThreadLocalSingleton单例

    public class ThreadLocalSingleton{
         private static final ThradLocal<ThreadLocalSingleton> threadLocalInstance = new ThreadLocal<ThreadLocalSingleton>(){
             @Override
             protected ThreadLocalSingleton initialValue() {
                 return new ThreadLocalSingleton();
             }
         }
        public static ThreadLocalSingleton getInstance() {return threadLocalInstance.get()}
    }
    // 此种单例常用于数据库连接池中,线程之间是隔离。

    总结:单例模式的特点:

                1. 私有化构造器
                   为什么要私有化呢,如果构造器是public修饰,那完全可以通过new来实例化该类的对象。这样其他处的代码就无法通                  过调用该类的构造函数来实例化该类的对象,只有通过该类提供的静态方法来得到该类的唯一实例。

               2. 保证线程安全
                    在多线程情况下,保证原子性。像饿汉式,枚举单例,双重检查,内部类式单例等都是线程安全的。

                3. 延迟加载
                   保证没有被使用之前是不会加载内存中的,

                4. 防止序列化和反序列化破坏单列

                5. 防御反射攻击单列

    那么class的生命周期一般来说会经历加载、连接、初始化、使用、和卸载五个阶段:参考: https://www.jianshu.com/p/9f369a17d1fb

    单例设计模式是23种设计模式常见的模式,也是我们熟知的模式。

    此文章也参考了: 
                             https://juejin.im/post/5b50b0dd6fb9a04f932ff53f
                             https://cloud.tencent.com/developer/article/1177048
                             https://www.jianshu.com/p/9f369a17d1fb
    文章中还有几处知识点还没有具体整理出来。
             

    相关文章

      网友评论

          本文标题:单例模式

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