深入单例模式

1. 饿汉模式

饿坏了,上来就先实例化一个对象,好处是代码简单,坏处是这个对象后面如果一直用不到,就是个浪费。

public class A{

private static A a = new A();

private A(){}

public static A getInstance(){

return a;

}

}

2. 懒汉模式

懒洋洋的,用到的时候才抱佛脚。最简单的懒汉模式如下,存在线程安全问题。

public static class A{

private static A a=null;

private A(){}

public static A getInstance(){

if(a==null){

a=new A();

}

return a;

}

}

终极解决办法是使用DCL(double check lock)加volatile,如下所示:

public static class A{

private static volatile A a=null;

private A(){}

public static A getInstance(){

if(a==null){

synchronized (A.class){

if(a==null){

a=new A();

}

}

}

return a;

}

}

  • DCL其实就是一锁二判三更新(在一锁之前加一个判断,提高并发时的效率,不需要每次一上来都先锁住消耗性能)。
  • 加volatile的目的是防止第8行发生指令重排。

第8行在底层会有3个操作:

  1. 分配一块内存
  2. 初始化这块内存(一般是调用类的构造函数)
  3. 将这块内存赋给变量(如上代码中是变量a)

如果在变量a不加volatile,上面的步骤可能发生指令重排变成1、3、2,导致a先等于一块没有初始化(2还未执行)或初始化一半(2执行了一半)的内存,这时候如果代码其他部分(不需要获取A.class的锁的部分)访问了a变量,就会读到预料外的值。

这有点类似于数据库里的"脏读",在事务处理中的数据被该事务外读到。

相关推荐
蝎子莱莱爱打怪25 分钟前
OpenClaw 从零配置指南:接入飞书 + 常用命令 + 原理图解
java·后端·ai编程
狼爷2 小时前
Go 没有 override?别硬套继承!用接口+嵌入,写更清爽的“覆盖”逻辑
java·go
小兔崽子去哪了4 小时前
Java 自动化部署
java·后端
ma_king4 小时前
入门 java 和 数据库
java·数据库·后端
后端AI实验室5 小时前
我用Cursor开发了3个月,整理出这套提效4倍的工作流
java·ai
码路飞9 小时前
GPT-5.3 Instant 终于学会好好说话了,顺手对比了下同天发布的 Gemini 3.1 Flash-Lite
java·javascript
SimonKing9 小时前
OpenCode AI编程助手如何添加Skills,优化项目!
java·后端·程序员
Seven9711 小时前
剑指offer-80、⼆叉树中和为某⼀值的路径(二)
java
怒放吧德德1 天前
Netty 4.2 入门指南:从概念到第一个程序
java·后端·netty
雨中飘荡的记忆1 天前
大流量下库存扣减的数据库瓶颈:Redis分片缓存解决方案
java·redis·后端