0

0

什么是Java中的内存模型(JMM)_工作内存与主内存的交互关系

P粉602998670

P粉602998670

发布时间:2026-03-17 10:13:31

|

997人浏览过

|

来源于php中文网

原创

Java内存模型(JMM)是定义多线程下可见性、有序性、原子性的抽象规范,划分主内存与工作内存逻辑概念,通过happens-before规则约束执行顺序,而非描述JVM物理内存结构。

什么是java中的内存模型(jmm)_工作内存与主内存的交互关系

Java内存模型不是JVM内存结构

很多人一看到“Java内存模型”就去翻JVM堆、栈、方法区的图,这是错的。JMM(Java Memory Model)是抽象规范,不描述物理内存布局,只定义volatilesynchronizedfinal等关键字在多线程下如何保证可见性、有序性、原子性。

它把内存划分为「主内存」和「工作内存」两个逻辑概念:

  • 主内存:所有线程共享,变量初始值和最终结果都存在这里
  • 工作内存:每个线程私有,线程操作变量前必须先从主内存拷贝副本到自己的工作内存中

注意:这里的“工作内存”不是JVM运行时数据区里的任何一块真实内存,它可能对应寄存器、CPU缓存,甚至部分在栈帧里——具体实现由JVM决定,但行为必须符合JMM约束。

变量读写必须经过主内存中转

线程不能直接读写主内存中的变量,所有操作都要走“读取→拷贝→计算→写回”流程。比如对一个普通int变量做i++,实际包含三步:

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

  • 从主内存读取i的值,加载到当前线程的工作内存
  • 在工作内存中执行加1操作
  • 把结果写回主内存

这三步不是原子的。如果两个线程同时执行i++,很可能都读到旧值,各自加1后写回,导致最终只+1而不是+2。

解决办法不是靠“避免并发”,而是用JMM提供的同步机制强制刷新:

  • synchronized块包裹操作:进入时清空工作内存,从主内存重新读;退出时把修改强制刷回主内存
  • 声明变量为volatile:每次读都从主内存取,每次写都立即刷回主内存(但不保证复合操作原子性)

happens-before规则才是JMM落地的关键

你写的代码顺序 ≠ 实际执行顺序。编译器、JVM、CPU都可能重排序,只要不破坏单线程语义。而JMM通过happens-before规则来约束多线程下的可见性边界。

AI改图神器
AI改图神器

AI万能图片编辑器,一键抠图,去水印,智能图片美化,照片转漫画,照片变活转视频,图片无损放大,一键背景虚化,位图智能转矢量图

下载

常见happens-before关系包括:

  • 程序次序规则:同一个线程内,前面的语句happens-before后面的语句(注意:不是字节码顺序,而是控制流意义上的先后)
  • 监视器锁规则:对同一个锁,unlock操作happens-before后续对该锁的lock操作
  • volatile变量规则:对一个volatile变量的写操作happens-before后续对该变量的读操作
  • 线程启动规则:Thread.start()调用happens-before子线程的任意动作

没有happens-before关系的操作,就可能被重排,也可能看不到最新值。别指望“我先写了,你后读就一定看到”——没规则保障,就是没保障。

容易被忽略的陷阱:final字段的特殊语义

final不只是“不可变”,它在JMM中有明确的发布语义。构造器中对final字段的写入,与对象引用的赋值之间存在happens-before关系。

这意味着:只要对象是正确构造的(没发生this逃逸),其他线程通过合法途径拿到该对象引用,就一定能看到final字段的正确初始化值。

但要注意:

  • 这个保障仅限final字段本身,不递归延伸到其引用的对象内部状态
  • 如果构造过程中把this泄露出去(比如注册监听器、开新线程),上述保障失效
  • static final基本类型还有更早的初始化时机优化,但引用类型仍依赖类初始化过程的同步语义

JMM的复杂性不在概念多,而在“哪些地方隐含了同步契约”。写并发代码时,盯着变量怎么读写没用,得盯住有没有建立happens-before链。漏掉一环,就可能在高并发下跑出难以复现的问题。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

1071

2023.08.02

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

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

618

2024.08.29

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

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

335

2025.08.29

C++中int的含义
C++中int的含义

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

235

2025.08.29

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

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

77

2025.10.23

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

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

448

2023.07.18

堆和栈区别
堆和栈区别

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

606

2023.08.10

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

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

448

2023.07.18

Nginx跨平台安装实操指南:Windows、macOS与Linux环境快速搭建
Nginx跨平台安装实操指南:Windows、macOS与Linux环境快速搭建

本指南详解Nginx在Windows、macOS及Linux系统的安装全流程。涵盖官方包解压、Homebrew一键部署、APT/YUM源配置及Docker容器化方案。无论新手或开发者,均可快速搭建运行环境,掌握跨平台核心指令,为后续配置与调优奠定坚实基础。

10

2026.03.16

热门下载

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

精品课程

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

共23课时 | 4.5万人学习

C# 教程
C# 教程

共94课时 | 11.5万人学习

Java 教程
Java 教程

共578课时 | 83.2万人学习

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

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