从单例谈double-check必要性,多种单例各取所需

Posted 烟花散尽13141

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了从单例谈double-check必要性,多种单例各取所需相关的知识,希望对你有一定的参考价值。


theme: fancy

前言

  • 前面铺掉了那么多都是在讲原则,讲图例。很多同学可能都觉得和设计模式不是很搭边。虽说设计模式也是理论的东西,但是设计原则可能对我们理解而言更加的抽象。不过好在原则东西不是很多,后面我们就可以开始转讲设计模式了。
  • 我的思路是按照设计模式进行分类整理。期间穿插相关的知识进行扩展从而保证我们学习的更加的全面。在正式开始前我现在这里立个Flag。争取在20周内完成我们设计模式章节的内容。期间可能会有别的学习,20周争取吧
  • 相信单例模式是大家第一个使用到的设计模式吧。不管你怎么样,我第一个使用的就是单例模式。其实单例模式也是分很多种的【饿汉式】、【懒汉式】。如果在细分还有线程安全和线程不安全版本的。

饿汉式

  • 顾名思义饿汉式就是对类需求很迫切。从Java角度看就是类随着JVM启动就开始创建,不管你是否使用到只要JVM启动就会创建。
 public class SingleFactory
 
     private static Person person = new Person();
 ​
     private SingleFactory()
     
     
 ​
     public static Person getInstance()
     
         return person;
     
 
  • 上面这段代码就是饿汉式单例模式。通过这单代码我们也能够总结出单例模式的几个特点
  • 特点
    隐藏类的创建即外部无法进行创建
    内部初始化好一个完整的类
    提供一个可以访问到内部实例的方法,这里指的是getInstance

  • 单例模式特点还是很容易区分的。饿汉式感觉挺好的,那为什么后面还会出现懒汉式及其相关的变形呢?下面我们就来看看饿汉式有啥缺点吧。
  • 首先上面我们提到饿汉式的标志性特点就是随着JVM 的启动开始生成实例对象。这是优点同时也是缺点。大家应该都用过Mybatis等框架,这些框架为了加快我们程序的启动速度纷纷推出各种懒加载机制。
  • 何为懒加载呢?就是用到的时候再去初始化相关业务,将和启动不相关的部分抽离出去,这样启动速度自然就快了起来了。在回到饿汉式,你不管三七二十一就把我给创建了这无疑影响了我的程序启动速度。如果这个单例模式你使用了倒还好,假如启动之后压根就没用到这个单例模式的类,那我岂不是吃力不讨好。不仅浪费了时间还浪费了我的空间。
  • 所以说,处于对性能的考虑呢?还是建议大家不要使用饿汉式单例。但是,存在即是合理的,我们不能一棒子打死一堆人。具体场景具体对待吧XDM。

🐶变形1

 public class SingleFactory
 
     private static Person person ;
 ​
     static 
         person = new Person();
     
 ​
     private SingleFactory()
     
     
 ​
     public static Person getInstance()
     
         return person;
     
 
  • 咋一看好像和上面的没啥区别哦。仔细对比你就会发现我们这里并没有立刻创建Person这个类,而是放在静态代码块中初始化实例了。
  • 放在静态代码块和直接创建其实是一样的。都是通过类加载的方式来进行实例化的。基本同根同源没啥可说的 。
  • 关于Static关键字我们之前也有说过,他涉及到的是类加载的顺序。我们在类加载的最后阶段就是执行我们的静态代码块

懒汉式

 public class SingleFactory
 
     private static Person person = null;
 ​
     private SingleFactory()
     
     
 ​
     public static Person getInstance()
     
         try
         
             Thread.sleep(30);
         
         catch (InterruptedException e)
         
            e.printStackTrace();
         
         if(person==null)
             person=new Person();
         
         return person;
     
 
  • 懒汉式就是将我们的对象创建放在最后一刻进行创建。并不是跟随类加载的时候生成对象的,这样会造成一定程度的内存浪费。懒汉式更加的提高了内存的有效利用。在getInstance方法中我们在获取对象前判断是否已经生成过对象。如果没有在生成对象。这种行为俗称懒,所以叫做懒汉式单例模式

🐱变形1

  • 上面懒汉式单例中我加入了睡眠操作。这是因为我想模拟出他的缺点。上面这种方式在高并发的场景下并不能保证系统中仅有一个实例对象。
 public class SingleFactory
 
     private static Person person = null;
 ​
     private SingleFactory()
     
     
 ​
     public static Person getIstance()
     
         try
         
             Thread.sleep(30);
         
         catch (InterruptedException e)
         
             e.printStackTrace();
         
         synchronized (SingleFactory.class)
         
             if (person == null)
             
                 person = new Person();
             
         
         return person;
     
 
  • 只需要加一把锁,就能保证线性操作了。但是仔细想想难道这样就真的安全了吗。

double-check

  • 在多线程下安全的单例模式应该非double-check莫属了吧。
 public class OnFactory 
     private static volatile OnFactory onFactory;
 ​
     public static OnFactory getInstance() 
         if (null == onFactory) 
             synchronized (OnFactory.class) 
                 if (null == onFactory) 
                     onFactory = new OnFactory();
                 
             
         
         return onFactory;
     
 
  • 这段代码是之前咱们学习double-check和volatile的时候写过的一段代码。在这里我们不仅在锁前后都判断了而且还加上了volatile进行内存刷新。关于volatile需要的在主页中搜索关键词即可找到。这里仅需要知道一点volatile必须存在否则线程不安全。

以上是关于从单例谈double-check必要性,多种单例各取所需的主要内容,如果未能解决你的问题,请参考以下文章

从单例谈double-check必要性,多种单例各取所需

从单例谈double-check必要性,多种单例各取所需

从单例模式的Double-Check看指令重排

static实现单例的隐患

从单例模式看Vue单例组件

从单例模式开始