下面我们来谈一下双重检查锁定与延迟初始化
volatile解决重排序问题
在前面学习懒汉模式实现单例模式的时候,我们已经使用过下面的这一套优化流程了
- 代码块中加锁判断单例对象是否已经初始化
- 如果已经初始化,直接返回单例对象
- 再进行判断多一次单例对象是否已经初始化,来判断单例模式对象是否已经初始化,因为可能同时有多个线程判断出单例对象未初始化,这时上一把锁,让一个线程进去初始化了,初始化了之后,应该再让其他线程再判断一次,看前面一个线程初始化没有(感觉这个方案可以解决一下缓存雪崩)
- 如果已经初始化,返回单例对象
- 给单例对象加volatile修饰,防止其构造指令出现重排序
代码如下(instance记得要被volatile修饰)

类初始化解决重排序问题
上面使用volatile可以解决重排序问题,在这里也是可以用类来解决重排序问题的
JVM在类的初始化阶段时,即在Class被加载后,且正在被线程使用之前,会执行类的初始化(初始化静态变量),在执行类的初始化期间,JVM会去获取一个锁,这个锁可以同步多个线程对同一个类的初始化,总的来说,就是利用类的初始化这个机制,让实例变量初始化的时候可以发生重排序,但其他线程看不到这个重排序,必须要等待完成整个类初始化过程才可以被访问这个类
还是以单例模式为例
要用类初始化来实现单例模式,其实就是使用静态内部类

本文探讨了懒汉模式中的双重检查锁定与延迟初始化技术,分析了使用volatile关键字及类初始化方式解决重排序问题的方法,并对比了两种方案的优缺点。
最低0.47元/天 解锁文章
168万+

被折叠的 条评论
为什么被折叠?



