如何在 Java 中使用 AtomicInteger 实现无锁的线程安全计数
如何在 Ja va 中使用 AtomicInteger 实现无锁的线程安全计数

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈
先来看一个核心的技术论断:AtomicInteger的incrementAndGet通常比synchronized快,因为它基于CPU的CAS指令,避免了阻塞和上下文切换的开销。但事情总有另一面:在高争用场景下,它可能因频繁重试反而效率更低,并且它只适用于简单的原子操作。
AtomicInteger 的 incrementAndGet 为什么比 synchronized 快
关键在于底层机制。incrementAndGet直接调用了CPU的CAS(比较并交换)指令,这是一种非阻塞的乐观策略。线程不会进入阻塞队列,自然也就绕开了上下文切换和锁竞争带来的性能损耗。相比之下,synchronized在高并发压力下,可能会经历偏向锁、轻量级锁到重量级锁的升级过程。一旦膨胀为系统级的互斥锁,性能就会出现断崖式下跌。
不过,这里必须划个重点:CAS并非万能钥匙。在极端的高争用场景下——想象一下上千个线程反复争夺同一个AtomicInteger——CAS操作失败和重试的次数会急剧增加,CPU消耗可能不降反升,这时候它的表现甚至可能不如一把简单的锁。
- 适用场景:计数器、序列号生成、统计指标(如请求量、错误数)这类典型的“读多写少”或“写操作本身很简单”的场景。
- 不适用场景:需要原子性地执行多个变量联动更新的复杂操作(例如“从余额减100的同时记录一条流水”)。这种时候,就该考虑
Lock或事务机制了。 - 一个小提醒:
incrementAndGet()返回的是增加后的新值,而getAndIncrement()返回的是增加前的旧值。在条件判断等逻辑中,可别用反了。
compareAndSet 是唯一能做条件更新的原子操作
如果想实现“仅当当前值为某个特定值时才进行更新”,那么compareAndSet是唯一正确的选择。千万别试图先get()再set()——这两个操作之间的间隙就是一个竞态窗口,根本不是原子的。
来看一个典型的错误写法:
int cur = counter.get();
if (cur == 5) {
counter.set(6); // ❌ 危险!执行get()后,cur的值可能已经被其他线程修改了
}
正确的做法应该是这样:
int expected = 5; boolean updated = counter.compareAndSet(expected, expected + 1); // updated为true表示更新成功;为false则表示在此期间值已被改动,需要决定重试或放弃
compareAndSet是典型的乐观锁策略:假设冲突很少发生,失败了就重试。它适合低到中等争用的场景。- 如果业务逻辑本身就很复杂,或者重试的代价很高,那么硬套CAS可能得不偿失,不如直接使用
ReentrantLock。 - 注意参数顺序:
compareAndSet(expectedValue, newValue),千万别把期望值和新值的位置写反了。
AtomicInteger 不能替代 long 类型的原子运算
这是一个容易踩坑的地方。AtomicInteger包装的是int类型(32位),其最大值是Integer.MAX_VALUE(2147483647)。一旦计数超过这个值,incrementAndGet()不会抛出异常,而是会发生静默的整数溢出,从最大值翻转到最小值(-2147483648)。
如果你的计数器有超过21亿的可能(比如全局日志行数、海量消息处理量),就必须换用其他方案:
AtomicLong:支持64位长整型,上限约9×10¹⁸,足以应对绝大多数场景。LongAdder:在超高并发的累加场景下,它的性能通常比AtomicLong更优。其内部采用了分段累加的策略来减少CAS争用。但需要注意的是,它不支持compareAndSet操作。- 切记,不要自己用
synchronized包裹一个long变量来模拟原子性,这等于放弃了无锁编程的全部优势。
get() 和 lazySet() 的内存语义差异常被忽略
get()是一个volatile读,它能保证线程总是能读到最新的值。而lazySet()(可以看作是set()的一个弱化版本)只保证写入操作本身不会被指令重排序,但并不保证这个新值能立即被后续的读操作看到——JVM可能会延迟将其刷新回主内存。
这意味着:
- 在写密集、且读写操作没有紧密耦合的场景下(例如信号量清零、设置一个状态标记),用
lazySet(value)替代set(value)可以略微提升写性能。 - 但是,它绝对不能用在对写后立刻读有强依赖的逻辑中。例如:
counter.lazySet(0); assert counter.get() == 0;这个断言是有可能失败的。 - 对于大多数业务代码而言,
lazySet可能根本用不上。除非你在明确的性能压测中发现set()成为了瓶颈,并且能够接受最终一致性的语义。
说到底,无锁编程并非没有成本,它只是将同步的成本从线程的阻塞和切换,转移到了CPU的重试循环和内存屏障上。因此,最关键的一步,是清醒地评估你的计数场景是否真的需要无锁方案。很多号称“高并发”的系统,其核心计数器的更新频率可能每秒只有几百次,在这种量级下,使用synchronized不仅完全够用,而且代码更直观,更不容易出错。
游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。
同类文章
使用Python合并与拆分Excel单元格的实用方法
使用Python合并与拆分Excel单元格的实用方法 处理Excel表格时,合并单元格是个绕不开的操作。无论是为了制作清晰美观的表头,还是为了突出显示某些关键信息,这个功能都相当实用。不过,当需要批量处理或者将流程自动化时,手动在Excel里点点划划就有点力不从心了。今天,我们就来聊聊如何用Pyth
SpringBoot OpenFeign整合okHttpClient实践
前言 在SpringCloud微服务架构中,服务间的数据传输,OpenFeign无疑是那个既简单又好用的选择。不过,它默认使用的客户端是JDK自带的HttpURLConnection,这里有个小细节值得注意:这个客户端本身并不具备连接池功能。 这意味着什么?简单来说,每一次发起远程调用,系统都会尝试
修改JAR文件并重新打包的两种方式
本文介绍两种修改 JAR 包内文件(如配置文件或 Class 文件)后重新打包的方式:Ja va 命令方式 与 Ant 脚本方式。 核心警告 对于 Spring Boot 的可执行 JAR 包,重新打包时严禁使用压缩(必须使用存储模式),否则会导致 ClassNotFoundException 或启
C++中INI配置文件读取技术详解
一、INI文件格式概述 在众多配置文件格式中,INI(Initialization)格式堪称经典。它以纯文本形式存储,结构清晰直观,既便于开发者手动编辑与维护,也易于程序进行自动化解析与读取。这种简单高效的特点,使其在软件配置、游戏设置、系统参数管理等场景中,至今仍被广泛应用。 1 1 基本结构 一
idea如何保存当前已修改的文件|恢复到未修改状态
1、打开git,如下步骤1 先来看第一张图,这是整个操作的起点。 在步骤2的区域,你会看到所有被修改过的文件都列在这里,一目了然。 而步骤3指向的代码区域,正是我们修改后还在报错的部分,问题就出在这儿。 这里有个关键细节:注意看圈4标识的地方,你所有修改过的代码行,IDE都会用淡绿色的背景高亮显示,
- 日榜
- 周榜
- 月榜
1
2
3
4
5
6
7
8
9
10
相关攻略
2015-03-10 11:25
2015-03-10 11:05
2021-08-04 13:30
2015-03-10 11:22
2015-03-10 12:39
2022-05-16 18:57
2025-05-23 13:43
2025-05-23 14:01
热门教程
- 游戏攻略
- 安卓教程
- 苹果教程
- 电脑教程
热门话题

