0

0

什么是Java并发中的非阻塞同步(Non-blocking Synchronization)_基于硬件指令取代系统调用的效率

P粉602998670

P粉602998670

发布时间:2026-03-15 07:17:26

|

688人浏览过

|

来源于php中文网

原创

非阻塞同步依靠CPU原子指令(如CAS)在用户态实现,无锁、无挂起、无中间态;典型代表是AtomicInteger.incrementAndGet(),需手动循环重试并每次重新读取最新值,推荐使用VarHandle而非Unsafe。

什么是java并发中的非阻塞同步(non-blocking synchronization)_基于硬件指令取代系统调用的效率

非阻塞同步到底靠什么实现

它不依赖 synchronizedReentrantLock 这类需要内核介入的锁机制,而是直接用 CPU 提供的原子指令(比如 CASLL/SC)在用户态完成状态变更。关键在于:所有操作要么全成功,要么完全不生效,没有中间态,也不挂起线程。

常见错误现象是误以为“不用 synchronized 就是非阻塞”——其实 volatile 读写只是保证可见性,不构成同步;而 AtomicInteger.incrementAndGet() 才是典型非阻塞同步,背后调的是 Unsafe.compareAndSwapInt()

  • 使用场景:高竞争但临界区极短的操作,比如计数器、无锁队列节点入队
  • 参数差异:AtomicReference.compareAndSet(expected, updated) 要求你显式传入期望值,失败后需自行重试,不是“自动重试”
  • 性能影响:避免了线程上下文切换开销,但在强竞争下可能因反复 CAS 失败导致 CPU 空转(spin)

CAS 循环重试怎么写才不出错

手动写 CAS 循环时,最容易漏掉对返回值的判断,或者把期望值写成固定值而不是每次从变量里重新读取。

正确做法是用 get() 获取当前值 → 计算新值 → compareAndSet(oldValue, newValue),失败就重来。JDK 的 AtomicInteger.getAndUpdate() 内部就是这么干的。

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

吉卜力风格图片在线生成
吉卜力风格图片在线生成

将图片转换为吉卜力艺术风格的作品

下载
  • 常见错误:写成 if (cas(old, new)) break; 却没更新 old,导致无限失败
  • 必须每次循环都调用 get()getOpaque() 重新读取最新值,不能复用上一轮的局部变量
  • 注意 ABA 问题:值从 A→B→A,CAS 会误判成功;必要时用 AtomicStampedReference 带版本戳

为什么 varHandle.compareAndSet()Unsafe 更推荐

Unsafe 是 JDK 内部 API,不同版本行为不一致,且从 JDK 9 开始默认禁止反射调用;VarHandle 是官方公开的、标准化的底层原子操作接口,语义清晰,还能跨平台适配不同 CPU 的内存序。

比如在 x86 上 VarHandle.compareAndSet() 编译为 lock cmpxchg,在 ARM 上则用 ldxr/stxr 序列,你不用操心。

  • 兼容性影响:JDK 9+ 中 Unsafe.compareAndSwap* 已被标记为 @Deprecated(since="9", forRemoval=true)
  • 性能差异几乎为零,但 VarHandle 支持更细粒度的内存语义控制,如 weakCompareAndSetPlain()
  • 初始化成本略高(需 MethodHandles.lookup().findVarHandle()),但只做一次,不影响运行时

非阻塞结构真比锁快?看这三点再决定

不是所有场景都适合非阻塞。盲目替换 synchronized 可能更慢,甚至引发活锁。

真正值得上非阻塞的地方,得同时满足:临界区代码少于 10 行、单次操作耗时远小于一次线程调度(通常

  • 容易踩的坑:用 ConcurrentLinkedQueue 替代 ArrayBlockingQueue,结果发现吞吐没涨反降——因为前者无锁但指针跳转多,缓存不友好
  • 调试困难:CAS 失败不会抛异常,只能靠日志或 JFR 观察 sun.misc.Unsafe::compareAndSwap* 调用频次
  • 复杂点往往不在算法本身,而在内存可见性边界和重排序约束,比如忘了用 VarHandle.setRelease() 保证后续写入不被重排到 CAS 之前

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
if什么意思
if什么意思

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

847

2023.08.22

java中break的作用
java中break的作用

本专题整合了java中break的用法教程,阅读专题下面的文章了解更多详细内容。

120

2025.10.15

java break和continue
java break和continue

本专题整合了java break和continue的区别相关内容,阅读专题下面的文章了解更多详细内容。

261

2025.10.24

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

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

76

2025.10.23

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1973

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

658

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2406

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

48

2026.01.19

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

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

49

2026.03.13

热门下载

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

精品课程

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

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.3万人学习

Java 教程
Java 教程

共578课时 | 82.1万人学习

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

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