0

0

Java面试之volatile关键字的内存可见性

畫卷琴夢

畫卷琴夢

发布时间:2026-01-21 08:58:25

|

674人浏览过

|

来源于php中文网

原创

volatile通过强制主内存读写和内存屏障保证可见性,但不保证原子性;它建立happens-before关系,使写前操作对后续读线程可见,依赖实际读写动作而非自动同步。

java面试之volatile关键字的内存可见性

volatile 为什么能保证内存可见性

因为 volatile 强制线程每次读取变量都从主内存重新加载,每次写入都立即刷新回主内存,绕过了 CPU 缓存的本地副本。JVM 在生成字节码时,会对 volatile 读写插入内存屏障(Memory Barrier):读操作前加 LoadLoadLoadStore,写操作后加 StoreStoreStoreLoad,阻止指令重排序并确保可见性传播。

注意:它不提供原子性——i++ 这种复合操作即使作用于 volatile int i,依然可能丢失更新。

volatile 不能替代 synchronized 的典型场景

当需要「读-改-写」原子性时,volatile 失效。比如计数器自增、状态标志配合业务逻辑判断后再操作等。

  • volatile boolean flag = false; 可用于通知线程退出,但若写成 if (flag) doSomething(); flag = true;,就存在竞态——两线程同时通过 if 判断后都执行 doSomething()
  • volatile int count; 无法安全执行 count++,因为底层是 getfield → iconst_1 → iadd → putfield 三步,中间可能被其他线程打断
  • 对象引用虽用 volatile 修饰,但其内部字段修改仍不具可见性,例如 volatile List<string> list = new ArrayList();</string>,后续 list.add("a") 不会触发对其他线程的可见保障

volatile 与 happens-before 规则的直接关联

JMM 中,对一个 volatile 变量的写操作,happens-before 于任意后续对该变量的读操作。这是唯一一条由 Java 语言规范明确定义的、不依赖锁的 happens-before 关系。

Android 本地数据存储 中文WORD版
Android 本地数据存储 中文WORD版

本文档主要讲述的是Android 本地数据存储;对于需要跨应用程序执行期间或生命期而维护重要信息的应用程序来说,能够在移动设备上本地存储数据是一种非常关键的功能。作为一名开发人员,您经常需要存储诸如用户首选项或应用程序配置之类的信息。您还必须根据一些特征(比如访问可见性)决定是否需要涉及内部或外部存储器,或者是否需要处理更复杂的、结构化的数据类型。跟随本文学习 Android 数据存储 API,具体来讲就是首选项、SQLite 和内部及外部内存 API。希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以

下载

立即学习Java免费学习笔记(深入)”;

这意味着:如果线程 A 写了 volatile boolean ready = true;,线程 B 后续读到 ready == true,那么线程 A 在写 ready 之前的所有内存操作(包括非 volatile 字段赋值),对线程 B 也一定可见。

class VolatileExample {
    int a = 0;
    volatile boolean flag = false;

    public void writer() {
        a = 1;           // 非 volatile 写
        flag = true;     // volatile 写 → 建立 happens-before 边界
    }

    public void reader() {
        if (flag) {      // volatile 读
            System.out.println(a); // 此处一定能打印 1
        }
    }
}

常见误判:volatile 能防止指令重排序,但不等于“顺序执行”

它只禁止特定类型的重排序(编译器和处理器不会把 volatile 读/写与前后某些操作乱序),但不保证多线程下所有语句的全局执行顺序一致。比如两个线程分别写不同的 volatile 变量,它们之间的相对顺序对第三方线程而言仍是不确定的。

真正容易被忽略的是:volatile 的可见性保障,依赖于「至少有一个线程执行了 volatile 写,另一个线程执行了 volatile 读」。如果读线程从未读取该变量,或者写线程写完后读线程才启动且未触发 volatile 读,那之前的写操作对其不可见——不是“一写全知”,而是“读到才同步”。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java中boolean的用法
java中boolean的用法

在Java中,boolean是一种基本数据类型,它只有两个可能的值:true和false。boolean类型经常用于条件测试,比如进行比较或者检查某个条件是否满足。想了解更多java中boolean的相关内容,可以阅读本专题下面的文章。

366

2023.11.13

java boolean类型
java boolean类型

本专题整合了java中boolean类型相关教程,阅读专题下面的文章了解更多详细内容。

42

2025.11.30

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

846

2023.08.22

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

203

2023.11.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1010

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

608

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

334

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

4

2026.03.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.1万人学习

Java 教程
Java 教程

共578课时 | 80.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号