0

0

Java中的JMM内存可见性问题演练_通过代码复现线程缓存失效

P粉602998670

P粉602998670

发布时间:2026-02-23 13:31:02

|

530人浏览过

|

来源于php中文网

原创

volatile不能解决所有可见性问题,因其不保证复合操作原子性,如i++仍会丢失更新;适用状态标志位,禁用于计数器;thread.sleep无法替代内存屏障;synchronized通过happens-before保障可见性;unsafe绕过jmm风险极高,业务代码禁用。

java中的jmm内存可见性问题演练_通过代码复现线程缓存失效

为什么volatile不能解决所有可见性问题 volatile只保证变量读写不被重排序、每次读都从主内存取、每次写都立即刷回主内存,但它**不保证复合操作的原子性**。比如i++(读-改-写三步),即使ivolatile,多个线程仍可能同时读到旧值,导致结果丢失。 常见错误现象:count字段用volatile修饰,循环1000次count++,最终结果却小于1000。 使用场景:适合状态标志位(如running)、单次写入多次读取的配置项;不适合计数器、累加器等需原子更新的场景。

实操建议:

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

SpeechEasy
SpeechEasy

SpeechEasy是一种合成语音解决方案,可以让用户从文本生成高质量、易于理解的音频。

下载
  • 确认是否真需要“可见性”而非“原子性”——如果涉及读+改+写,优先考虑AtomicIntegersynchronized
  • 不要对volatile字段做条件轮询+修改(如if (flag) { flag = false; doWork(); }),这仍是竞态
  • 注意JVM版本差异:Java 5+才完全支持volatile语义,旧版本有弱实现

Thread.sleep(1)复现缓存不一致有多危险 很多人用Thread.sleep(1)在循环里“等另一个线程更新”,以为能触发缓存同步,其实它只是让出CPU,**不强制刷新本地缓存,也不建立happens-before关系**。结果就是:主线程永远看不到子线程对ready的修改,陷入无限等待。 常见错误现象:子线程设ready = true后退出,主线程while(!ready)死循环,即使等几秒也不退出。 性能影响:Thread.sleep引入不可控延迟,且无法替代内存屏障;在高并发下反而放大可见性问题。

实操建议:

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

  • 别用sleep代替同步机制——它不是内存栅栏,也不是通知手段
  • 若真要等待状态变更,用wait/notifyCountDownLatchAtomicBoolean配合get()轮询(后者需配合volatile语义)
  • 调试时可用Unsafe.fullFence()手动插内存屏障,但生产环境禁用

synchronized块为何能修复可见性却常被误用 synchronized不仅互斥,还隐式建立happens-before:退出同步块时,会把本地缓存刷回主内存;进入同步块时,会清空本地缓存并从主内存重新读取。这是JMM规定的语义,不是JVM优化开关。 容易踩的坑:锁对象不一致。比如一个线程锁this,另一个锁new Object(),或锁不同实例,就完全不构成同步。 使用场景:适用于需要原子性+可见性的临界区,尤其是共享状态多字段联动更新(如balancelastUpdate需一起变)。

实操建议:

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

  • 锁对象必须是所有线程都能访问到的同一个引用,推荐用私有final Object lock = new Object()
  • 避免锁thisgetClass(),防止外部代码意外获取同一把锁导致阻塞
  • 同步块粒度要小——只包真正共享操作,别把System.out.println或网络调用裹进去

Unsafe绕过JMM验证可见性是否靠谱 UnsafegetLongVolatileputLongVolatile等方法可对任意内存地址施加volatile语义,常被用来在无锁数据结构中模拟volatile字段行为。但它**绕过了编译器和JVM的常规检查**,极易因地址错位、类型不匹配导致崩溃或静默错误。 常见错误现象:用Unsafe.objectFieldOffset获取字段偏移失败(字段被优化掉或访问权限不足),后续getXXXVolatile返回随机值。 兼容性风险:JDK 9+模块化后Unsafe默认不可达;JDK 17开始彻底移除公开构造方式;GraalVM等非HotSpot VM可能无对应实现。

实操建议:

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

  • 仅限深入理解JMM或开发底层框架时使用,业务代码绝对禁止
  • 必须配合try/catch捕获RuntimeExceptionUnsafe方法不声明异常但会抛)
  • 字段偏移务必用Unsafe.staticFieldOffsetobjectFieldOffset动态获取,硬编码数字在不同JVM版本上必然失效
JMM的可见性不是“缓存有没有刷新”的硬件问题,而是“什么时候必须刷新”的语义契约。最易被忽略的是:没有同步动作(volatile写、synchronized进出、Lock操作、线程启动/终止等),编译器和CPU就有权永远不刷新——哪怕你用System.out.println打了一百行日志。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

828

2023.08.22

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

199

2023.11.20

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

103

2023.09.25

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

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

71

2025.10.23

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

543

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

27

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

39

2026.01.06

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

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

715

2023.08.10

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

1030

2026.02.13

热门下载

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

精品课程

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

共23课时 | 3.8万人学习

C# 教程
C# 教程

共94课时 | 10万人学习

Java 教程
Java 教程

共578课时 | 70.4万人学习

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

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