0

0

在Java中Java内存模型如何理解_Java并发底层解析

P粉602998670

P粉602998670

发布时间:2026-02-11 03:03:32

|

865人浏览过

|

来源于php中文网

原创

Java内存模型(JMM)是定义多线程下变量可见性、有序性及重排序规则的抽象规范,非物理内存布局;其核心为“主内存-工作内存”模型,通过volatile、synchronized和final提供不同强度的内存语义保障。

在java中java内存模型如何理解_java并发底层解析

Java内存模型(JMM)不是物理内存布局

很多人一看到“内存模型”就下意识联想到堆、栈、方法区这些运行时数据区——这是常见误解。JMM 不描述 JVM 实际怎么分配内存,而是定义了多线程环境下,变量读写操作 在不同线程间如何可见、何时有序、哪些重排序被允许。它的核心是抽象的“主内存-工作内存”模型,每个线程有自己的工作内存(可理解为寄存器或CPU缓存的抽象),所有共享变量都存在于主内存中。

关键点在于:线程对变量的所有操作(读、写、加锁、解锁)都必须通过工作内存与主内存交互,不能直接读写主内存。这就引出了可见性、原子性、有序性三大问题。

volatile 关键字到底禁止了什么重排序?

volatile 是 JMM 最轻量级的同步机制,但它只保证两点:变量对所有线程的可见性,以及禁止特定类型的指令重排序。它不保证复合操作的原子性(比如 i++)。

编译器和处理器对 volatile 变量的读写有明确约束:

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

  • 写 volatile 变量前,其工作内存中所有变量的值必须刷新回主内存(相当于插入一个 StoreStore + StoreLoad 屏障)
  • 读 volatile 变量后,其工作内存中所有变量的值必须从主内存重新加载(相当于插入一个 LoadLoad + LoadStore 屏障)
  • 禁止对 volatile 读/写与普通读/写做重排序(但 volatile 读与 volatile 读之间、volatile 写与 volatile 写之间仍可能重排)

例如:volatile boolean flag = false; 后续的 int x = 10; 不会被重排到 flag = true 之前——这正是 DCL(双重检查锁定)中 instance = new Singleton() 需要 volatile 的根本原因。

synchronized 和 happens-before 规则怎么对应?

synchronized 的语义远不止“加锁”。在 JMM 中,它通过 happens-before 规则建立线程间的偏序关系:一个线程对锁的 unlock 操作 happens-before 另一个线程对该锁的 lock 操作。

ChatGPT Website Builder
ChatGPT Website Builder

ChatGPT网站生成器,AI对话快速生成网站

下载

这意味着:

  • 退出同步块前,该线程工作内存中所有变量的修改都会刷新到主内存
  • 进入同步块时,该线程会清空工作内存,并从主内存重新读取所有共享变量(包括非 volatile 的)
  • 所以 synchronized 天然解决可见性、原子性和有序性三类问题

注意:synchronized 块的锁对象必须是同一个实例才有意义;锁的粒度影响性能,但不影响 JMM 语义本身。另外,wait()/notify() 也参与 happens-before 链——notify() happens-before 被唤醒线程的 wait() 返回。

final 字段的内存语义常被忽略

final 字段不只是“不可变”,它在构造完成那一刻就向其他线程提供了安全发布保障。JMM 规定:如果一个对象被正确构造(即构造过程中 this 引用未逸出),那么其他线程看到该对象的 final 字段时,一定能看到构造器中为其设置的值,且不会看到默认值(如 0、null)。

这个保障依赖于两个动作的禁止重排序:

  • 禁止把 final 字段的写入重排序到构造器结束之后
  • 禁止把初次读取 final 字段重排序到构造器开始之前

换句话说,只要没发生 this 逃逸,final int x = 42; 的写入对其他线程就是立即可见的——不需要 volatile 或 synchronized。这是实现无锁不可变对象(如 StringLocalDateTime)的底层基础。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

688

2023.08.02

java中boolean的用法
java中boolean的用法

在Java中,boolean是一种基本数据类型,它只有两个可能的值:true和false。boolean类型经常用于条件测试,比如进行比较或者检查某个条件是否满足。想了解更多java中boolean的相关内容,可以阅读本专题下面的文章。

357

2023.11.13

java boolean类型
java boolean类型

本专题整合了java中boolean类型相关教程,阅读专题下面的文章了解更多详细内容。

36

2025.11.30

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

243

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

685

2024.03.01

string转int
string转int

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

688

2023.08.02

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

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

557

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

193

2025.08.29

2026春节习俗大全
2026春节习俗大全

本专题整合了2026春节习俗大全,阅读专题下面的文章了解更多详细内容。

68

2026.02.11

热门下载

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

精品课程

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

共23课时 | 3.5万人学习

C# 教程
C# 教程

共94课时 | 9.2万人学习

Java 教程
Java 教程

共578课时 | 63.4万人学习

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

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