为什么 synchronized 不能防止指令重排序?

在某乎看到一个提问,大家讨论 synchronized 能不能防止指令冲排序,咋说的都有,我发现大家学习底层技术很多时候会有误区。

先说我的观点:synchronized 绝对不能防止它内部代码的指令重排序!

下边说说我的分析哈,不对的大家讨论

synchronized 到底防了什么?

很多人觉得 synchronized 能防重排序,是因为看到了它能保证有序性这句话。但这句话是有大前提的!

打个比方:synchronized 就像是一间带锁的单人洗手间。

并发视角的有序性,线程 A 进去了,把门反锁;线程 B 只能在门外排队。A 出来之后,B 才能进去。线程 B 看来,A 在里面的所有动作是一次性做完的,这就叫保证了有序性和原子性。

但是!线程 A 关上门之后,在洗手间里到底是先脱裤子再上厕所,还是先脱衣服再洗脸?CPU 和编译器为了追求极致的执行效率,是完全可能把 A 在洗手间里的动作顺序打乱的(指令重排序)。

只要 A 在洗手间里的瞎搞不影响最终结果,CPU 就觉得没毛病。synchronized 根本管不住 CPU 在单线程内部的微操。

双重检查锁DCL为什么要加volatile

我们来看看经典的 DCL 单例是怎么写的:

csharp 复制代码
public class Singleton {
    // 注意:这里如果不加 volatile,将引发灾难
    privatestatic Singleton instance; 

    public static Singleton getInstance() {
        if (instance == null) { // <--- 致命的第一重检查(在锁外面)
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton(); // <--- 万恶之源
                }
            }
        }
        return instance;
    }
}

这段代码看着没啥毛病,外层判断空,避开锁的性能开销。内层加锁,保证只有一个线程去 new 对象。坑就坑在 instance = new Singleton() 这句代码上。

因为 Java 字节码CPU 执行层面,他就不是一个原子操作,它被分成三步:

1、分配内存空间

2、初始化对象,执行构造函数

3、将 instance 引用指向分配的内存地址

正常人的逻辑是 1 -> 2 -> 3。

但是!前面说了哈,synchronized 管不住洗手间里面的重排序。CPU 一看,步骤 2 和 3 互相不依赖啊,为了效率,我给你优化成 1 -> 3 -> 2 吧!。

灾难发生的全过程

假设这时候发生了 1 -> 3 -> 2 的重排序:

  • 线程 A 进到了 synchronized 块里,开始执行 new Singleton()。
  • 线程 A 执行了步骤 1,然后执行了步骤 3。注意,此时对象还没执行构造函数,但 instance 已经不是 null 了!
  • 就在这极其致命的瞬间,线程 B 杀过来了。线程 B 执行第一句代码 if (instance == null)。注意这句代码是在 synchronized 外面的!
  • 线程 B 根本不需要等锁!线程 B 看到 instance != null,对象 new 好了,直接 return instance 拿去用。
  • 结果线程 B 拿到的是一个还没执行构造函数的半成品对象,线程 B 调用里面的成员变量,直接爆出 NullPointerException 或者拿到错乱的初始值,系统当场崩溃。

发现问题了吗?

synchronized 确实能把线程 A 锁在里面,但它防不住线程 B 在外面偷看!因为 DCL 最大的卖点就是第一层检查没有加锁!

这就是为什么必须给 instance 加上 volatile 关键字

arduino 复制代码
private static volatile Singleton instance;

volatile 的核心作用,除了大家熟知的保证内存可见性之外,在 DCL 这个场景下,最关键的作用是插入内存屏障,禁止指令重排序。

加了 volatile 之后,CPU 和编译器看到这个变量,就会老老实实地立正站好。它强制要求必须先完全执行完步骤 1 和 2 ,才能执行步骤 3。

这样一来,只要线程 B 看到 instance != null,那么这个对象绝对是可用的,不会拿到半成品。

相关推荐
AMoon丶2 小时前
Golang--锁
linux·开发语言·数据结构·后端·算法·golang·mutex
神奇小汤圆2 小时前
Java面试被问:跟我讲下JVM和JMM?
后端
李日灐2 小时前
改造红黑树实现封装 map/set:感受C++ 标准容器的精妙设计与底层实现
开发语言·数据结构·c++·后端·算法·红黑树
傲文博一2 小时前
SFTP 端口 是多少?为什么连接不上?一篇讲清楚
后端
李日灐2 小时前
【优选算法1】双指针经典算法题
数据结构·c++·后端·算法·刷题·双指针
别看我只是一直狼2 小时前
🚀 程序员高效 Prompt 实战速查手册
前端·后端
技术钻石流2 小时前
面向“传统程序员”的端到端 10x Vibe Coding 指南(大型需求) - 从面向业务开发转向面向“Agent 员工”开发
前端·后端·ai编程
xiao_juzi3 小时前
OpenClaw 环境变量配置完全指南
后端
2501_921649493 小时前
免费港股实时行情 API:功能、性能与接入指南
开发语言·后端·python·金融·restful