0

0

在Java里volatile关键字如何使用_Javavolatile可见性原理说明

P粉602998670

P粉602998670

发布时间:2026-01-18 16:48:55

|

659人浏览过

|

来源于php中文网

原创

volatile仅解决变量可见性,不保证原子性;适用于单写多读且写不依赖当前值的场景,如状态标志位;禁止指令重排序,但不保证long/double在32位JVM上的原子写;不延伸至引用对象内部。

在java里volatile关键字如何使用_javavolatile可见性原理说明

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 不保证这三步整体原子。

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

✅ 正确做法:用 AtomicInteger 或加锁。

volatile 如何禁止指令重排序

JVM 和 CPU 可能对指令重排以优化性能,但 volatile 写操作会插入“StoreStore”屏障,读操作插入“LoadLoad”和“LoadStore”屏障,确保:

AMiner
AMiner

AMiner——新一代智能型科技情报挖掘与服务系统,能够为你提供查找论文、理解论文、分析论文、写作论文四位一体一站式服务。

下载
  • 写 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 位变量(longdouble)的读写拆成两个 32 位操作,即所谓“非原子性更新”。虽然现代 64 位 JVM 默认已保证原子性,但若目标环境明确是 32 位 JVM(如某些嵌入式或旧 Android 版本),且变量未声明为 volatile,就可能出现“半个 long 值”的问题。

所以:如果变量是 longdouble,又需跨线程安全读写,必须加 volatile(或用 AtomicLong 等)。

注意:volatile 在这里的作用不是“让写变原子”,而是强制使用原子的读写指令,并禁用重排序——JVM 对 volatile long 的读写总是当作单个不可分割的操作来实现。

真正容易被忽略的是:volatile 的内存语义只作用于它修饰的变量本身,不延伸到该变量引用的对象内部。比如 volatile List list = new ArrayList();,只保证 list 引用的可见性,不保证 list.add() 的线程安全。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

837

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

741

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

736

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

72

2026.01.16

热门下载

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

精品课程

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

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 7万人学习

Java 教程
Java 教程

共578课时 | 47.5万人学习

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

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