volatile关键字与可见性原理深度解析
时间:2026-01-25 15:15:34 140浏览 收藏
一分耕耘,一分收获!既然都打开这篇《Java中volatile的使用与可见性原理详解》,就坚持看下去,学下去吧!本文主要会给大家讲到等等知识点,如果大家对本文有好的建议或者看到有不足之处,非常欢迎大家积极提出!在后续文章我会继续更新文章相关的内容,希望对大家都有所帮助!
volatile仅解决变量可见性,不保证原子性;适用于单写多读且写不依赖当前值的场景,如状态标志位;禁止指令重排序,但不保证long/double在32位JVM上的原子写;不延伸至引用对象内部。

volatile 不是用来替代 synchronized 或锁的,它只解决变量的可见性问题,不保证原子性。如果你需要“读-改-写”这类复合操作线程安全(比如 i++),volatile 无效。
volatile 变量必须满足“单次读或单次写”场景
它适用于一个线程写、多个线程读,且写操作不依赖当前值的场景。典型例子是状态标志位:
public class TaskRunner {
private volatile boolean running = true;
public void stop() {
running = false; // 单次写,无依赖
}
public void run() {
while (running) { // 多线程读,每次读都看到最新值
// do work
}
}
}
常见错误是误用于计数器:
❌ volatile int count = 0; 然后在多线程里执行 count++ —— 这会丢失更新,因为 count++ 包含读、加1、写三步,volatile 不保证这三步整体原子。
✅ 正确做法:用 AtomicInteger 或加锁。
volatile 如何禁止指令重排序
JVM 和 CPU 可能对指令重排以优化性能,但 volatile 写操作会插入“StoreStore”屏障,读操作插入“LoadLoad”和“LoadStore”屏障,确保:
- 写 volatile 变量前的所有普通写,不会被重排到该写之后
- 读 volatile 变量后的所有普通读/写,不会被重排到该读之前
这个特性常用于双重检查锁定(DCL)单例模式中防止对象未初始化完成就被其他线程看到:
public class Singleton {
private static volatile Singleton instance;
public static Singleton getInstance() {
if (instance == null) {
synchronized (Singleton.class) {
if (instance == null) {
instance = new Singleton(); // 非原子:分配内存 → 初始化 → 赋值给 instance
}
}
}
return instance;
}
}
没有 volatile,JVM 可能把“赋值给 instance”提前到“初始化”之前(只要语义合法),导致其他线程拿到一个未构造完的对象。
volatile 不能保证 long/double 的原子写(仅限 32 位 JVM)
Java 规范允许对 64 位变量(long、double)的读写拆成两个 32 位操作,即所谓“非原子性更新”。虽然现代 64 位 JVM 默认已保证原子性,但若目标环境明确是 32 位 JVM(如某些嵌入式或旧 Android 版本),且变量未声明为 volatile,就可能出现“半个 long 值”的问题。
所以:如果变量是 long 或 double,又需跨线程安全读写,必须加 volatile(或用 AtomicLong 等)。
注意:volatile 在这里的作用不是“让写变原子”,而是强制使用原子的读写指令,并禁用重排序——JVM 对 volatile long 的读写总是当作单个不可分割的操作来实现。
真正容易被忽略的是:volatile 的内存语义只作用于它修饰的变量本身,不延伸到该变量引用的对象内部。比如 volatile List,只保证 list 引用的可见性,不保证 list.add() 的线程安全。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《volatile关键字与可见性原理深度解析》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
163 收藏
-
147 收藏
-
158 收藏
-
210 收藏
-
173 收藏
-
267 收藏
-
383 收藏
-
186 收藏
-
126 收藏
-
311 收藏
-
488 收藏
-
225 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习