通俗理解单例模式-懒汉式双重校验锁

简单的单例模式:(懒汉式)

package com.zcp.juc.single;

/**
 * @author zcp
 * @description
 * @created by 2020-03-26 22:50
 */
public final class Singleton {

    private static Singleton INSTANCE=null;

    private Singleton(){

    }

    public static Singleton getInstance(){
        if(INSTANCE==null){
            INSTANCE=new Singleton();
        }
        return INSTANCE;
    }

}

通俗讲:单例=单个实例,即每次获取相同的对象实例,因为java中创建对象需要消耗资源,单例模式正好解决了对象的频繁创建。那么懒汉式是什么呢?懒呗,我就是不想那么早初始化,有需要我再初始化。

很多人都以为懒汉式写到这,可是在多线程环境下呢?上述代码完了吗?
没有!!!
问题来了:
有两个线程T1、T2,当线程T1执行到if条件判断时,发现INSTANCE==null,还没创建实例呢,T2线程也走到了这个if条件判断,发现INSTANCE==null,那么两条线程继续向下执行,就会导致new了两个对象,这显然不符合单例模式,不是我们想要的结果。

怎么办的?
在有可能发生问题的地方加锁,不知道在哪?没关系,把整个方法都加上锁

package com.zcp.juc.single;

/**
 * @author zcp
 * @description
 * @created by 2020-03-26 22:50
 */
public final class Singleton {

    private static Singleton INSTANCE=null;

    private Singleton(){

    }

    public static synchronized Singleton getInstance(){
        if(INSTANCE==null){
            INSTANCE=new Singleton();
        }
        return INSTANCE;
    }

}

在静态方法上加锁,相当于对类对象加锁

上述代码等价于

package com.zcp.juc.single;

/**
 * @author zcp
 * @description
 * @created by 2020-03-26 22:50
 */
public final class Singleton {

    private static Singleton INSTANCE=null;

    private Singleton(){

    }

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

线程安全问题解决了,还有什么问题呢?
问题来了:T1线程先拿到锁,T2线程阻塞,T1线程同步代码块执行完毕,成功创建了对象,释放了锁,此时T2线程拿到锁,再执行if判断,发现实例已经被初始化,大家不觉得很麻烦吗?为什么不直接告诉T2对象已经被创建了,直接获取就是了,还要加一次锁,大哥啊,加锁不要钱啊?
怎么办呢?
那么传说中双重判断来了,在加锁前判断一次INSTANCE是否等于null,不等于直接就返回实例了,这样也不用再加锁判断了。(也就是首次访问需要同步,而之后就没有synchronized了),这样做还有问题吗?
biao急啊?
go on..

package com.zcp.juc.single;

/**
 * @author zcp
 * @description
 * @created by 2020-03-26 22:50
 */
public final class Singleton {

    private static Singleton INSTANCE=null;

    private Singleton(){

    }

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

什么问题呢?
看字节码(重点看17~24行字节码指令)

 0: getstatic     #2                  // 获取静态变量INSTANCE
       3: ifnonnull     37           //判断INSTANCE是不是Null   如果不是Null就跳转执行37行
       6: ldc           #3                  // 获得了类对象 
       8: dup                              //把类对象的引用指针复制了一份
       9: astore_0                      //然后临时存储了复制的一份,是为了将来解锁用
      10: monitorenter              //开始执行同步代码块
      11: getstatic     #2                  // 拿到静态变量
      14: ifnonnull     27                  //如果不为Null,执行27行
      17: new           #3                  // 为Null,继续执行,创建对象,将对象的引用入栈
      20: dup                                // 复制一份这个对象的引用(引用地址)
      21: invokespecial #4                  // 利用对象的引用来调用构造方法(根据引用地址调用)
      24: putstatic     #2                  // 原来的这一份的引用对应赋值操作,把他赋值给静态变量
      27: aload_0                        //把临时存储的类对象取出来
      28: monitorexit                    //解锁,退出同步代码块
      29: goto          37                //跳转到37行
      32: astore_1
      33: aload_0
      34: monitorexit
      35: aload_1
      36: athrow
      37: getstatic     #2                  // 获取静态变量
      40: areturn                            //返回结果
    Exception table:
       from    to  target type
          11    29    32   any
          32    35    32   any

_
17: new #3 // 创建对象,将对象的引用入栈 new Singleton()
20: dup // 复制一份这个对象的引用(引用地址)
21: invokespecial #4 // 利用对象的引用来调用构造方法(根据引用地址调用)
24: putstatic #2 // 利用一份对象引用赋值给static Instance

jvm虚拟机在执行时有可能做优化(指令重排序优化),也就是可能先执行24,再执行21,那么会导致什么问题呢?synchronized只可能保证同步代码块内原子性(注:synchronized代码块内的代码仍可能发生有序性问题,即指令重排序),但是无法保证外面if判断。啥意思呢?

当T1线程执行到同步代码块内,发生了指令重排序,先调用了24行的指令,将对象的引用赋值给了static Instance,那么此时T2执行到同步代码块外面的if判断,就会发现Instance不为Null,就继续执行返回,可返回的时候,T1还未将构造方法初始完毕。

总结:

  • 关键在于0:getstatic在monitor外面,就好像不守规矩的人,他可以越过monitor读取Instance变量的值
  • T1还未完全将构造方法初始完毕,如果构造方法内要执行很多初始化操作,那么T2拿走的将是一个未初始化完毕的实例
  • 对Instance使用volatile修饰即可,可以禁止指令重排序。

最终完全的代码:

package com.zcp.juc.single;

/**
 * @author zcp
 * @description
 * @created by 2020-03-26 22:50
 */
public final class Singleton {

    private static volatile Singleton INSTANCE=null;

    private Singleton(){

    }

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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,128评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,316评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,737评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,283评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,384评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,458评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,467评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,251评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,688评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,980评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,155评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,818评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,492评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,142评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,382评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,020评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,044评论 2 352

推荐阅读更多精彩内容