JVM内存结构

JVM内存结构概述

JVM内存结构是Java虚拟机在执行Java程序时划分的多个内存区域,每个区域有特定用途。主要包括堆、方法区、虚拟机栈、本地方法栈和程序计数器。

堆(Heap)

堆是JVM中最大的一块内存区域,被所有线程共享。用于存放对象实例和数组。堆分为新生代(Young Generation)和老年代(Old Generation),新生代进一步分为Eden区、Survivor区(From和To)。堆是垃圾回收的主要区域。

  • 新生代:新创建的对象首先分配在Eden区,经过Minor GC后存活的对象移动到Survivor区。
  • 老年代:长期存活的对象最终会晋升到老年代,当老年代空间不足时触发Full GC。

方法区(Method Area)

方法区也是线程共享的内存区域,存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等。在HotSpot虚拟机中,方法区的实现称为"永久代"(PermGen),但在JDK 8后被"元空间"(Metaspace)取代,元空间使用本地内存而非JVM内存。

虚拟机栈(Java Virtual Machine Stacks)

每个线程私有的内存区域,生命周期与线程相同。虚拟机栈存储栈帧(Stack Frame),每个方法调用会创建一个栈帧,用于存储局部变量表、操作数栈、动态链接和方法出口信息。

  • 局部变量表:存储方法参数和局部变量,基本数据类型直接存储值,对象类型存储引用。
  • 操作数栈:用于执行字节码指令的临时数据存储。
  • 动态链接:指向运行时常量池的方法引用。
  • 方法出口:记录方法返回时的位置。

本地方法栈(Native Method Stack)

与虚拟机栈类似,但为本地方法(Native Method)服务。本地方法栈的具体实现依赖于虚拟机设计,有些虚拟机(如HotSpot)将虚拟机栈和本地方法栈合二为一。

程序计数器(Program Counter Register)

线程私有的小内存空间,记录当前线程执行的字节码指令地址。如果执行的是本地方法,程序计数器的值为空(Undefined)。程序计数器是唯一一个在JVM规范中没有规定任何OutOfMemoryError情况的区域。

直接内存(Direct Memory)

虽然不是JVM规范定义的内存区域,但直接内存(如NIO使用的堆外内存)可以通过Native方法直接分配,受本机总内存限制。直接内存的读写速度通常优于堆内存,但管理和回收需要额外注意。

内存溢出与调优

  • 堆溢出OutOfMemoryError: Java heap space,可通过调整-Xms-Xmx参数增加堆大小。
  • 栈溢出StackOverflowError,通常由递归调用过深引起,可通过-Xss调整栈大小。
  • 方法区溢出OutOfMemoryError: Metaspace,可通过-XX:MetaspaceSize-XX:MaxMetaspaceSize调整元空间大小。

示例参数

bash 复制代码
# 设置堆初始大小和最大大小
-Xms256m -Xmx1024m
# 设置元空间大小
-XX:MetaspaceSize=128m -XX:MaxMetaspaceSize=512m
# 设置线程栈大小
-Xss256k
相关推荐
han_hanker25 分钟前
@Validated @Valid 用法
java·spring boot
小CC吃豆子26 分钟前
详细介绍一下静态分析工具 SonarQube
java
DevOpenClub30 分钟前
全国三甲医院主体信息 API 接口
java·大数据·数据库
上海合宙LuatOS35 分钟前
LuatOS扩展库API——【exremotecam】网络摄像头控制
开发语言·网络·物联网·lua·luatos
言慢行善37 分钟前
SpringBoot中的注解介绍
java·spring boot·后端
一勺菠萝丶39 分钟前
管理后台使用手册在线预览与首次登录引导弹窗实现
java·前端·数据库
feng_you_ying_li41 分钟前
C++11,{}的初始化情况与左右值及其引用
开发语言·数据结构·c++
xiaotao1311 小时前
JS new 操作符完整执行过程
开发语言·前端·javascript·原型模式
TE-茶叶蛋1 小时前
结合登录页-PHP基础知识点解析
android·开发语言·php
无巧不成书02181 小时前
Java包(package)全解:从定义、使用到避坑,新手零基础入门到实战
java·开发语言·package·java包