当前位置: 首页 > news >正文

JVM 内存区域详解

JVM 内存区域详解

Java 虚拟机(JVM)的内存区域划分为多个部分,每个部分有特定的用途和管理机制。以下是 JVM 内存区域的核心组成及其功能:

一、运行时数据区(Runtime Data Areas)

1. 线程共享区域

内存区域别名特性异常类型
堆(Heap)新生代+老年代存储对象实例和数组,GC主要工作区域OutOfMemoryError
方法区元空间(Metaspace)存储类信息、常量、静态变量、JIT编译后的代码(JDK8+使用本地内存实现)OutOfMemoryError

2. 线程私有区域

内存区域特性异常类型
程序计数器(PC Register)记录当前线程执行的字节码行号,唯一不会OOM的区域
虚拟机栈(JVM Stack)存储栈帧(局部变量表、操作数栈、动态链接、方法出口)StackOverflowError
本地方法栈(Native Stack)为Native方法服务StackOverflowError

二、各区域深度解析

1. 堆内存(Heap)

  • 新生代 (Young Generation)

    • Eden区:对象初次分配区域
    • Survivor区(S0/S1):Minor GC后存活对象暂存区
    • 比例:默认Eden:S0:S1 = 8:1:1(可通过-XX:SurvivorRatio调整)
  • 老年代 (Old Generation)

    • 存储长期存活对象(默认经过15次GC仍存活的对象)
    • 触发Major GC/Full GC
  • 配置参数

    -Xms1024m  # 初始堆大小
    -Xmx1024m  # 最大堆大小
    -XX:NewRatio=2  # 老年代/新生代比例
    

2. 方法区(Method Area)

  • 存储内容

    • 类型信息(类名、访问修饰符等)
    • 运行时常量池
    • 静态变量(JDK7+静态变量移至堆中)
    • JIT编译后的代码缓存
  • 演进历史

    • JDK7及之前:永久代(PermGen),-XX:PermSize/-XX:MaxPermSize
    • JDK8+:元空间(Metaspace),使用本地内存,-XX:MetaspaceSize/-XX:MaxMetaspaceSize

3. 虚拟机栈(JVM Stack)

  • 栈帧结构

    栈帧
    局部变量表
    操作数栈
    动态链接
    方法返回地址
  • 局部变量表

    • 基本数据类型直接存储值
    • 引用类型存储指向堆的引用
    • 槽位(Slot)是基本存储单位(32位)
  • 配置参数

    -Xss256k  # 设置线程栈大小
    

三、直接内存(Direct Memory)

  • 特点

    • 不属于JVM运行时数据区
    • 通过NIO的ByteBuffer.allocateDirect()分配
    • 避免Java堆与Native堆间数据拷贝
  • 相关异常

    • OutOfMemoryError: Direct buffer memory
  • 配置参数

    -XX:MaxDirectMemorySize=256m
    

四、内存区域交互关系

线程1
程序计数器
虚拟机栈
本地方法栈
线程2
程序计数器
虚拟机栈
本地方法栈
对象实例
方法区
类信息
运行时常量池
本地方法接口

五、异常示例与调优

1. 堆内存溢出

// 持续创建大对象
List<byte[]> list = new ArrayList<>();
while(true) {
    list.add(new byte[1024*1024]); // 1MB per object
}

调优:增大-Xmx,分析内存泄漏

2. 栈溢出

// 无限递归
public void stackOverflow() {
    stackOverflow();
}

调优:增大-Xss或修复递归终止条件

3. 方法区溢出

// 借助CGLib持续生成类
while(true) {
    Enhancer enhancer = new Enhancer();
    enhancer.setSuperclass(OOMObject.class);
    enhancer.setUseCache(false);
    enhancer.create();
}

调优:增大-XX:MaxMetaspaceSize

六、JDK工具监控

工具功能示例命令
jstat监控堆内存和GC情况jstat -gcutil <pid> 1000
jmap堆转储和分析jmap -heap <pid>
VisualVM图形化监控所有内存区域可视化工具
NMT(Native Memory Tracking)跟踪本地内存使用-XX:NativeMemoryTracking=detail

七、重要注意事项

  1. JDK版本差异

    • JDK7:字符串常量池在方法区(PermGen)
    • JDK8+:字符串常量池移至堆中
  2. 对象创建流程

    类加载检查 → 分配内存(堆)→ 初始化 → 设置对象头 → 执行<init>方法
    
  3. 内存分配策略

    • 优先在Eden区分配
    • 大对象直接进入老年代(-XX:PretenureSizeThreshold
    • 长期存活对象进入老年代(-XX:MaxTenuringThreshold

理解JVM内存区域是性能调优和故障诊断的基础,合理配置各区域大小可以显著提升应用稳定性和性能。

http://www.dtcms.com/a/113352.html

相关文章:

  • 01人工智能基础入门
  • JavaWeb 课堂笔记 —— 01 HTML
  • AutoCAD2026中文版下载安装教程
  • GESP:2025-3月等级8-T1-上学
  • Java异步编程中的CompletableFuture介绍、常见错误及最佳实践
  • 多周期多场景的供应链优化问题 python 代码
  • QMainWindow添加状态栏
  • 【深度学习】嘿马深度学习目标检测教程第2篇:目标检测算法原理,3.2 R-CNN【附代码文档】
  • 【C/C++算法】蓝桥杯之递归算法(如何编写想出递归写法)
  • 2025 年 4 月补丁星期二预测:微软将推出更多 AI 安全功能
  • Java实现N皇后问题的双路径探索:递归回溯与迭代回溯算法详解
  • 【微机及接口技术】- 第四章 内部存储器及其接口(中)
  • LlamaIndex实现RAG增强:上下文增强检索/重排序
  • 我是如何写作的?
  • LintCode第974题-求矩阵各节点的最短路径(以0为标准)
  • 如何将本地更改的README文件同步到自己的GitHub项目仓库
  • OmniParser: 让大模型化身“电脑管家”
  • 洛谷 P3214 [HNOI2011] 卡农
  • 2.IO流的体系和字节输出流FileOutputStream的基本用法
  • macos 魔搭 模型下载 Wan-AI ComfyUI
  • L2-024 部落 #GPLT,并查集 C++
  • 智能驾驶中预测模块简介
  • 广州t11基地顺利完成交割,TCL华星技术产能双升级
  • 【java】Class.newInstance()
  • 硬币找零问题
  • 特征值与特征向量:从理论到应用的全面解析
  • Java类加载问题
  • STM32单片机入门学习——第16节: [6-4] PWM驱动LED呼吸灯PWM驱动舵机PWM驱动直流电机
  • 《AI大模型应知应会100篇》第4篇:Transformer架构深入浅出:大模型的基石
  • cadence17.4和16.6同时安装无法使用的问题