0

0

在Java里Atomic类解决了什么问题_Java原子操作类库解析

P粉602998670

P粉602998670

发布时间:2026-01-23 09:02:02

|

105人浏览过

|

来源于php中文网

原创

AtomicInteger等类解决单变量“读-改-写”竞态问题,不替代锁或解决所有并发问题;i++非原子致丢失更新,volatile无法修复;CAS实现原子操作,但不支持多变量事务或复杂条件逻辑。

在java里atomic类解决了什么问题_java原子操作类库解析

Java 的 AtomicIntegerAtomicLong 等类,解决的是**多线程环境下对单个变量做“读-改-写”操作时的竞态条件问题**,而不是替代锁或解决所有并发问题。

为什么 i++ 在多线程下会出错

看似简单的 i++ 实际包含三步:读取当前值 → 加 1 → 写回内存。若两个线程同时执行,可能都读到旧值(比如 5),各自加 1 后都写回 6,结果丢失一次更新。

这种非原子性行为无法靠 volatile 修复——它只保证可见性,不保证操作的原子性。

  • volatile int i = 0; 不能防止 i++ 的中间步骤被穿插
  • AtomicInteger i = new AtomicInteger(0);i.incrementAndGet() 是底层用 CPU 的 CAS(Compare-And-Swap)指令实现的单指令原子操作
  • 不是所有平台都支持 CAS,但 HotSpot JVM 在 x86/ARM 上已通过 lock cmpxchgldxr/stxr 等指令兜底

Atomic 类能做什么,不能做什么

它们适用于「单变量、无依赖、无副作用」的原子更新场景,比如计数器、序列号生成、状态标志位切换。

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

一旦涉及多个变量联动(如转账:A 减、B 增),或需要复合逻辑判断(如“只有余额 > 100 才扣款”),AtomicXxx 就力不从心了——它不提供条件式多步原子操作,也没有内置重试机制。

  • ✅ 支持:自增、自减、CAS 设置、getAndSet、getAndUpdate、updateAndGet
  • ❌ 不支持:跨字段事务、阻塞等待、公平性控制、超时重试
  • ⚠️ 注意:getAndUpdateupdateAndGet 接收的 UnaryOperator 函数体里不应有 I/O、锁或耗时操作,否则会卡住 CAS 自旋

AtomicReference 与对象引用的“原子性”边界

AtomicReference 保证的是「引用本身」的读写原子性,不是它指向的对象内部状态的线程安全。

比如用 AtomicReference<list>></list> 存一个 ArrayList,你用 compareAndSet 替换整个引用没问题;但若调用 get().add("x"),这个 add 操作仍需额外同步——因为 ArrayList 本身不是线程安全的。

  • 常见误用:atomicRef.get().size() 返回后,对象可能已被其他线程修改,结果不可靠
  • 正确姿势:尽量把状态封装进不可变对象,再用 AtomicReference 做整体替换
  • 复杂状态变更建议配合 StampedLockReadWriteLock,而非强行拆成多个 Atomic 字段

性能开销和 ABA 问题的真实影响

CAS 自旋在低竞争时比锁快很多,但高竞争下可能反复失败重试,反而更耗 CPU。JVM 对 AtomicXxx 有专门优化(如内联、消除冗余屏障),所以不要自己用 Unsafe 模拟。

ABA 问题确实存在:变量从 A→B→A,CAS 误判为“没变过”,但实际上中间状态已改变。但现实中绝大多数场景(计数器、开关标志)根本不在乎中间是否绕过 B;真需要严格版本控制的,应使用 AtomicStampedReference 或直接上 LongAdder / ConcurrentHashMap 等更高层工具

  • 除非你在写无锁/队列这类底层结构,否则基本不用操心 ABA
  • LongAdder 在高并发计数场景比 AtomicLong 更高效,但它不提供精确的实时值(最终一致性)
  • 别为了“看起来更原子”而滥用 AtomicBoolean 控制流程——有时一个 synchronized 块更清晰、更易维护

真正难的从来不是选哪个 Atomic 类,而是判断「这里到底需不需要原子性」,以及「原子性边界划在哪」——变量级?对象级?还是业务语义级?

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

1031

2023.08.02

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

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

613

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

c++中volatile关键字的作用
c++中volatile关键字的作用

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

76

2025.10.23

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

605

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

765

2023.08.10

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

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

26

2026.03.13

热门下载

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

精品课程

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

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.3万人学习

Java 教程
Java 教程

共578课时 | 81.7万人学习

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

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