DCL 实现中:
1、为什么需要使用两个 if 语句?
2、为什么使用了 synchronized 关键字还需要使用 volatile 关键字?
3、双重校验锁使用需要注意的问题
public class SingletonDemo {
private static Object object = null;
}
public class SingletonDemo {
private static Object object = null;
// 初始化和获取实例
public Object getObject() {
synchronized (SingletonDemo.class) {
if (object == null) {
object = new Object();
}
return object;
}
}
}
public class SingletonDemo {
private volatile static Object object = null;
public Object getObject() {
// 如果实例已经初始化完成,直接返回实例不获取锁
if (object != null){
return object;
}
synchronized (SingletonDemo.class) {
if (object == null) {
object = new Object();
}
return object;
}
}
}
创建一个对象分为初始化和实例化两部分,大致可以分为以下几步:
1、在堆中申请一份内存
2、创建对象
3、将 object 指向我们对象的内存引用
如果没有指令重排的情况下,我们拿到的对象一定是完整的对象,但是可能存在指令重排优化,上面的顺序可能变成下面这样:
1、申请一份内存
2、将 object 指向我们对象的内存引用
3、创建对象
那么我们将会拿到一个没有实例化完成的对象,因此我们需要禁止指令重排,Java 提供了 volatile 指令来禁止指令重排。
为了提升执行速度/性能,计算机在执行程序代码的时候,会对指令进行重排序。什么是指令重排?简单来说就是系统在执行代码的时候并不一定是按照程序的代码的顺序依次执行。
指令重排可以保证单线程串行语义一致(as-if-serial),但是没有义务保证多线程间的语义也一致,所以在多线程下,指令重排可能会导致一些问题。
关于指令重排更多内容可以参考 一文读懂 Java Memory Model(JMM)
最后,我们得到了终极版本的代码:
public class SingletonDemo {
private volatile static Object object = null;
public Object getObject() {
// 如果实例已经初始化完成,直接返回实例不获取锁
if (object != null){
return object;
}
synchronized (SingletonDemo.class) {
if (object == null) {
object = new Object();
}
return object;
}
}
}
你好,我是 Lorin 洛林,一位 Java 后端技术开发者!座右铭:Technology has the power to make the world a better place.
我对技术的热情是我不断学习和分享的动力。我的博客是一个关于Java生态系统、后端开发和最新技术趋势的地方。
作为一个 Java 后端技术爱好者,我不仅热衷于探索语言的新特性和技术的深度,还热衷于分享我的见解和最佳实践。我相信知识的分享和社区合作可以帮助我们共同成长。
在我的博客上,你将找到关于Java核心概念、JVM 底层技术、常用框架如Spring和Mybatis 、MySQL等数据库管理、RabbitMQ、Rocketmq等消息中间件、性能优化等内容的深入文章。我也将分享一些编程技巧和解决问题的方法,以帮助你更好地掌握Java编程。
我鼓励互动和建立社区,因此请留下你的问题、建议或主题请求,让我知道你感兴趣的内容。我期待与你一起在技术之路上前进,一起探讨技术世界的无限可能性。
保持关注我的博客,让我们共同追求技术卓越。