0

0

Java中的volatile能保证原子性吗_i++操作失效的剖析与正确并发累加姿势

P粉602998670

P粉602998670

发布时间:2026-03-15 16:21:01

|

891人浏览过

|

来源于php中文网

原创

volatile不能保证i++的原子性,因其仅确保可见性和禁止重排序,而i++包含读—改—写三步,中间可能被其他线程打断。

java中的volatile能保证原子性吗_i++操作失效的剖析与正确并发累加姿势

volatile 不能保证 i++ 的原子性

直接说结论:volatile 只保证可见性和禁止指令重排序,不保证复合操作的原子性。i++ 看似简单,实则是“读—改—写”三步:先读取 i 的当前值,再加 1,最后写回内存。哪怕 ivolatile 修饰的,这三步之间仍可能被其他线程打断。

常见错误现象是:多线程下对 volatile int i = 0 执行 1000 次 i++,最终结果远小于 1000 —— 这不是偶发 bug,是必然发生。

  • i++ 在字节码层面至少对应三条指令:getstaticiaddputstatic
  • volatile 仅确保每次 getstaticputstatic 都直通主存,但不锁住中间过程
  • 两个线程同时读到 i == 5,各自加 1 后都写回 6,一次累加就丢了

为什么 AtomicInteger.incrementAndGet() 能行

它用的是 CAS(Compare-And-Swap)+ 自旋,把“读—改—写”封装成一个不可分割的硬件级操作。JVM 会将其编译为类似 lock xadd 的 CPU 指令(x86 平台),由处理器保证执行期间不会被抢占。

使用场景明确:需要在多线程中安全修改单个整型计数器,且不希望引入重量级锁(如 synchronized)时,AtomicInteger 是首选。

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

  • 不是所有方法都原子:AtomicInteger.get()set() 是,但 getAndIncrement()incrementAndGet() 才真正对应 i++++i
  • CAS 在高竞争下会自旋重试,极端情况下可能比 synchronized 更耗 CPU
  • 注意其 lazySet() 方法:写操作不立即刷主存,适合做“只写不读”的标记位(如关闭标志)

synchronized 做累加也完全可行,但要注意粒度

很多人以为 AtomicInteger 是唯一解,其实用 synchronized 同样能正确实现并发累加,关键是锁的范围要小、对象要稳。

In3D
In3D

把真人变成化身,创建逼真且可自定义的虚拟角色

下载

容易踩的坑是锁了不该锁的东西,比如每次新建一个 Object 当锁,或者锁了 this 却在不同实例上调用 —— 根本没起到互斥作用。

  • 推荐用私有 final 锁对象:private final Object lock = new Object();,避免外部干扰
  • 不要锁字符串字面量或 Class 对象,除非你清楚类加载和字符串常量池的共享逻辑
  • 如果累加只是某个方法内的局部行为,且该方法本身已同步,就不必额外加锁

别忽略 long/double 的特殊性:volatile 也不保原子读写

Java 规范允许 JVM 将 64 位的 longdouble 的读写拆成两个 32 位操作(尤其在 32 位 JVM 上)。这意味着即使加了 volatile,也可能出现“半个值被更新、另半个还是旧值”的撕裂现象(torn write)。

所以,只要涉及多线程读写 longdouble,无论是否 volatile,都得用 AtomicLong / AtomicLongFieldUpdater,或者用 synchronized 包裹读写逻辑。

  • volatile long timestamp 在某些老 JVM 上,可能被读出一个高位来自旧值、低位来自新值的“拼接数”
  • AtomicLong 内部同样依赖 CAS,且对 64 位操作做了平台适配,是唯一可靠选择
  • 现代 64 位 JVM 通常默认原子处理 long/double,但规范不保证,生产环境不能赌

真正难的不是选 AtomicInteger 还是 synchronized,而是想清楚:这个累加是否真的需要跨线程可见?是否和其他状态变更存在耦合?一旦涉及多个变量协同更新,volatile 和原子类都无能为力,只能靠锁或更高级的并发结构。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1570

2023.10.24

字符串常量的表示方法
字符串常量的表示方法

字符串常量的表示方法:1、使用引号;2、转义字符;3、多行字符串;4、原始字符串;5、字符串连接;6、字符串字面量和对象;7、编码问题。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

145

2023.12.26

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1570

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

651

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1229

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1205

2024.04.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

69

2026.03.13

热门下载

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

精品课程

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

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.4万人学习

Java 教程
Java 教程

共578课时 | 82.7万人学习

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

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