0

0

什么是JVM的自旋锁与适应性自旋_避免线程频繁挂起与恢复的CPU空转策略

P粉602998670

P粉602998670

发布时间:2026-03-14 11:17:02

|

592人浏览过

|

来源于php中文网

原创

自旋锁本质是线程在用户态死循环检查锁是否释放,以CPU时间换调度开销;适用于临界区极短的多核场景,不适用I/O或复杂操作;JDK 6+默认启用适应性自旋,动态调整自旋次数。

什么是jvm的自旋锁与适应性自旋_避免线程频繁挂起与恢复的cpu空转策略

自旋锁到底在“空转”什么?

自旋锁不是真的让线程“睡觉”,而是让它在用户态死循环检查锁是否被释放——不进内核、不挂起、不切换上下文。本质是用 CPU 时间换线程调度开销。

常见错误现象:Thread.getState() 返回 RUNNABLE,但实际啥也没干,CPU 使用率却莫名飙升;GC 日志里没看到大量线程阻塞,但吞吐量上不去。

  • 只在多核机器上有效:单核 CPU 自旋 = 白白饿死其他线程
  • 适用场景:临界区极短(比如几十纳秒到几微秒),例如计数器递增、简单状态标记更新
  • 不适用场景:synchronized 块里有 I/O、sleep、数据库查询、复杂计算——这时自旋纯属浪费
  • JDK 6+ 默认开启,无需加 -XX:+UseSpinning;但默认最多自旋 10 次,超时即走传统阻塞流程

为什么需要“适应性”?固定 10 次太傻了

固定次数的自旋就像定闹钟:锁刚释放你就停了,或者锁早该放了你还硬等满 10 轮——这两种都亏。

适应性自旋让 JVM 记住每个锁的历史表现:上次在这把锁上自旋成功了?持有锁的线程还在跑?那这次就多等几轮;如果连续几次都失败,下次干脆跳过自旋,直接挂起。

Peppertype.ai
Peppertype.ai

高质量AI内容生成软件,它通过使用机器学习来理解用户的需求。

下载
  • 判断依据只有两个:前一次自旋是否成功 + 当前持有锁的线程是否仍在运行中(not in BLOCKED/WAITING)
  • 它不依赖任何 JVM 启动参数,完全由 HotSpot 运行时动态决策
  • 你无法用 -XX:PreBlockSpin 干预适应性行为——这个参数只影响“非适应性”模式下的初始值,JDK 6+ 已基本弃用
  • 注意:适应性只对同一对象实例上的锁生效;换一个对象,历史清零

怎么确认你的锁真在自旋?别靠猜

光看代码没法判断 synchronized 是否触发了自旋,得看运行时行为。最直接的方式是观察线程栈和锁竞争指标。

  • jstack <pid> 查看线程状态:如果看到大量线程卡在 java.lang.Thread.State: RUNNABLE,且堆栈停留在 synchronized 方法入口(如 at java.util.HashMap.put(HashMap.java:...)),同时 CPU 高,大概率正在自旋
  • 启用 JVM 锁统计:-XX:+PrintGCDetails -XX:+PrintSafepointStatistics -XX:PrintSafepointStatisticsCount=1,配合 jstat -printcompilation <pid> 观察 safepoint 停顿是否异常频繁(自旋失败后进入阻塞会触发 safepoint)
  • 不要依赖 VisualVM 或 JMC 的“线程 CPU 时间”排序——自旋线程显示为高 CPU,但未必是瓶颈;重点看它是否集中在某几个热点对象上

关掉自旋锁能解决问题吗?

不能。禁用自旋(-XX:-UseSpinning)只会让所有锁竞争立刻走重量级阻塞路径,反而可能放大性能问题。

  • 副作用明显:线程频繁进出内核态 → 上下文切换暴涨 → 缓存失效加剧 → 实际延迟可能比自旋还高
  • 真正该做的,是减少锁竞争本身:比如把大锁拆成小锁(分段锁)、用 java.util.concurrent 替代 synchronized、或改用无锁结构(AtomicInteger, ConcurrentHashMap
  • 如果发现某类对象上自旋失败率长期 >80%,说明它根本不适合自旋——这时应优先考虑消除锁(-XX:+EliminateLocks)或关闭偏向锁(-XX:-UseBiasedLocking)来加速轻量级锁路径

自旋锁不是开关,是权衡。它不解决锁设计问题,只缓解调度代价;真正要调的,永远是锁的粒度和持有时间,而不是自旋次数。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
堆和栈的区别
堆和栈的区别

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

447

2023.07.18

堆和栈区别
堆和栈区别

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

606

2023.08.10

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

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

447

2023.07.18

堆和栈区别
堆和栈区别

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

606

2023.08.10

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

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

766

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

100

2025.12.01

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

389

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2112

2023.08.14

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

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

26

2026.03.13

热门下载

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

精品课程

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

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