JVM详解——内存结构

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6

文章目录

内存结构

在这里插入图片描述

1、 运行时数据区

Java虚拟机在运行Java程序期间将管理的内存划分为不同的数据区不同的区域负责不同的职能有各自的生命周期这些区域统称为运行时数据区。

从线程私有和共享的角度区分

线程私有 程序计数器、Java虚拟机栈、本地方法栈

线程共享 堆、方法区

栈是运行时的单位而堆是存储的单位​。

栈解决程序的运行问题即程序如何执行或者说如何处理数据。堆解决的是数据存储的问题即数据怎么放、放在哪。


2、虚拟机栈

作用 虚拟机栈也称为Java栈主要管理Java程序的运行保存方法的局部变量、部分结果并参与方法的调用和返回。

每个线程都有自己的栈栈中的数据以栈帧的格式存在一个线程上正在执行的每个方法都有自己对应的一个栈帧方法调用栈帧压栈方法结束栈帧弹出。

栈帧的内部结构

  • 局部变量表

主要用于存储方法参数和定义在方法体内的局部变量包括基本数据类型和对象引用

  • 操作数栈

主要用于保存计算过程的中间结果同时作为计算过程中变量临时的存储空间在方法执行过程中根据字节码指令往操作数栈中写入数据或提取数据即入栈、出栈

  • 动态链接

指向运行时常量池的方法引用

在 Java 源文件被编译到字节码文件中时所有的变量和方法引用都作为符号引用保存在 Class 文件的常量池中动态链接的作用就是为了将这些符号引用转换为调用方法的直接引用

  • 方法返回地址

用来存放调用方法的PC寄存器的值

方法结束的两种方式正常执行、出现异常非正常退出

  • 一些附加信息

携带与Java虚拟机相关的一些附加信息比如对程序调试提供支持的信息。


3、本地方法栈

本地方法栈就是Java调用非Java代码的接口也就是Native Method​本地方法。

作用与操作系统交互Java应用有时候需要依赖一些底层系统的支持比如在拷贝数组时候使用系统拷贝。


4、程序计数器

程序计数器是一块较小的内存空间可以看作是当前线程所执行的字节码的行号指示器 存储指向下一条指令的地址即将执行的指令代码。

作用CPU在切换线程时JVM需要明确线程下一条应该执行什么字节码指令而PC程序计数器中存储的就是下一条指令的地址并且每个线程的执行进度不同不同的线程需要有自己的PC程序计数器。


5、 堆

堆是Java虚拟机管理的内存中最大的一块被所有线程共享。堆用来存放对象实例几乎所有的对象实例以及数据都在这里分配内存。

堆也是垃圾回收的主要区域为了高效的进行垃圾回收虚拟机把堆内存在逻辑上划分为三块区域

  • 新生代 存放新创建的对象和没达到一定年龄的对象
  • 老年代 被长时间使用的对象老年代内存更大
  • 元空间JDK1.8之前称永久代存放一些方法中的临时对象

Java 虚拟机规范规定Java 堆可以是处于物理上不连续的内存空间中只要逻辑上是连续的即可像磁盘空间一样。实现时既可以是固定大小也可以是可扩展的主流虚拟机都是可扩展的通过 -Xmx​ 和 -Xms​ 控制如果堆中没有完成实例分配并且堆无法再扩展时就会抛出 OutOfMemoryError​ 异常。

-Xmx: 堆的起始内存默认初始化大小为 电脑内存/64

-Xmx​堆的最大内存默认初始化大小为 电脑内存/4


6、方法区

方法区是 JVM 规范中定义的一个概念用于存储类信息、常量池、静态变量、JIT编译后的代码等数据永久代是Hotspot虚拟机对方法区的实现JDK8之后改为元空间

JDK8 之前使用永久代实现方法区容易内存溢出因为永久代有 -XX:MaxPermSize ​​上限即使不设置也有默认大小。JDK7 把放在永久代的字符串常量池、静态变量等移出保存到堆中JDK8 中取消永久代改用在本地内存中实现的元空间代替把 类型信息、字段、方法、常量保存在本地内存的元空间中。


7、运行时常量池

运行时常量池是方法区的一部分一个有效的class​字节码文件中除了包含类的版本信息、字段、方法以及接口等描述信息外还包含常量池表用于存放各种字面量和对类型、域和方法的符号引用这部分内容将在类加载后存放到方法区的运行时常量池中。

运行时常量池相对于 Class 文件常量池的另一个重要特征是动态性Java 语言并不要求常量一定只有编译期间才能产生运行期间也可以将新的常量放入池中。

比如·String· 类的 intern()​ 方法如果字符串常量池中存在对应得字面量方法返回该字面量得地址如果不存在则创建一个对应得字面量放入运行时常量池返回字面量地址。


8、内存溢出和内存泄漏

内存溢出OutOfMemory​程序申请的内存超过JVM能够提供的内存大小导致内存溢出主要为堆内存溢出。

内存泄漏 Memory Leak​无法释放已申请的内存虚拟机不能再次使用该内存


9、 堆溢出

堆用于存储对象实例只要不断创建对象并保证 GC Roots 到对象有可达路径避免垃圾回收随着对象数量的增加程序需要的内存空间就会超出JVM分配的内存空间导致OOM程序崩溃例如在 while 死循环中一直 new 创建实例。

出现堆OOM的情况

  • 堆的内存大小设置不当
  • 程序中存在内存泄漏问题或应用中有大量占用内存的对象并且无法及时释放

解决方案

  • -Xms -Xmx ​​修改堆的内存大小
  • 通过内存监控软件去查找程序中的泄漏代码

线上排查方式

  1. 获取内存的dump文件1. 配置JVM启动参数当触发了OOM异常时自动生成 2. 使用jmap工具生成
  2. 使用MAT工具分析dump文件
    • 如果是内存泄漏可以查看泄漏对象的GC Roots的引用链通过类信息和引用链信息定位到代码位置进行解决
    • 堆空间分配不足以满足业务需求提升堆内存空间


参考文章
‍1. https://pdai.tech/md/java/jvm/java-jvm-struct.html
2. https://blog.csdn.net/weixin_45629285/article/details/128050932

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6