JVM的内存模型

JVM内存详解
本文深入解析JVM内存结构,涵盖虚拟机栈、堆、方法区、程序计数器和本地方法栈的功能与管理机制,详细阐述堆内存的分代概念,包括新生代与老年代的划分,以及Eden区与Survivor区的角色。同时,探讨了Java栈、方法区、本地方法栈和程序计数器的工作原理,以及JVM内存参数设置。

一、JVM内存结构

JVM 内存共分为虚拟机栈,堆,方法区,程序计数器,本地方法栈五个部分。JVM的内存模型和类加载机制是分不开的 JVM类加载机制

在这里插入图片描述

二、JVM内存区域功能

  1. 堆内存
    堆内存是JVM内存模型中最大的一块区域,被所有线程共享,是在JVM启动时候进行创建的。几乎所有的对象的空间分配都是在堆内存上进行分配的。

    考虑到JVM的内存回收机制,堆内存可以划分为新生代和老年代两个区域(默认新生代与老年代的空间大小为1:2)。新生代可以再划分为Eden区、From Survivor区和To Survivor区(三者比例为8:1:1)。几乎所有的新对象的创建都是在Eden区进行的。在垃圾回收(GC)过程中,Eden中的活跃对象会被转移到Survivor区,当再到达一定的年龄(经历Minor GC15次),会被转移到老年代中。

    堆可以处于物理上不连续的内存空间中,但是需要满足逻辑上的连续。在实现时,可以实现成固定大小的, 也可以是可扩展的,不过当前主流的虚拟机都是按照可扩展来实现的

  2. 方法区
    方法区又被成为永久代(HotSpot虚拟机的设计团队选择把GC分代收集扩展至方法区),同样也是被所有的线程共享的。该区域中主要保存类的信息、静态变量、常量和即时编译器编译后的代码等数据。
    JDK1.7中,已经把放在永久代的字符串常量池移到堆中。JDK1.8撤销永久代,引入元数据区域。

    方法区不需要连续的内存,可以选择固定大小或者可扩展。并且还可以选择不实现垃圾收集。
    相对而言,垃圾收集行为在这个区域是比较少出现的,但并非数据进入了方法区就如永久代的名字一样“永久”存在了。这个区域的内存回收目标主要是针对常量池的回收和对类型的卸载,一般来说这个区域的回收“成绩”比较难以令人满意,尤其是类型的卸载,条件相当苛刻,但是这部分区域的回收确实是有必要的。当方法区无法满足内存分配需求时,将抛出OutOfMemoryError异常。
    在这里插入图片描述

    老年代:2/3的堆空间
    年轻代:1/3的堆空间
    eden区:8/10 的年轻代
    survivor0: 1/10 的年轻代
    survivor1:1/10的年轻代

  3. Java栈
    java栈是线程私有的内存区域,其中存储的是栈帧。对于一个java的方法,其开始调用,则会创建一个栈帧,保存到java栈中;当该方法执行完成,则对应的是出栈的过程。

    每个栈帧中,保存执行对应方法所必须的信息,主要包含局部变量表,操作数栈,动态链接和方法出口信息。
    在这里插入图片描述

  • 局部变量表:
    局部变量表中,可以存放的数据有8种基本数据类型(boolean,byte,char,short,int,float,long,double),对象引用和returnAddress类型。其中long和double因为是64位,会占用两个局部变量的空间。

    在Java虚拟机规范中,对这个区域规定了两种异常状况:如果线程请求的栈深度大于虚拟机所允许的深度(比如递归调用的时候),将抛出StackOverflowError异常;如果虚拟机栈可以动态扩展(当前大部分的Java虚拟机都可动态扩展,只不过Java虚拟机规范中也允许固定长度的虚拟机栈),当扩展时无法申请到足够的内存时会抛出OutOfMemoryError异常。

  • 操作数栈
    后进先出LIFO,最大深度由编译期确定。栈帧刚建立使,操作数栈为空,执行方法操作时,操作数栈用于存放JVM从局部变量表复制的常量或者变量,提供提取,及结果入栈,也用于存放调用方法需要的参数及接受方法返回的结果。
    操作数栈可以存放一个jvm中定义的任意数据类型的值。
    在任意时刻,操作数栈都一个固定的栈深度,基本类型除了long、double占用两个深度,其它占用一个深度

  • 动态连接:
    每个栈帧都包含一个指向运行时常量池中该栈帧所属方法的引用,持有这个引用是为了支持方法调用过程中的动态连接。Class文件的常量池中存在有大量的符号引用,字节码中的方法调用指令就以常量池中指向方法的符号引用为参数。这些符号引用,一部分会在类加载阶段或第一次使用的时候转化为直接引用(如final、static域等),称为静态解析,另一部分将在每一次的运行期间转化为直接引用,这部分称为动态连接。

  • 方法返回地址:
    当一个方法被执行后,有两种方式退出该方法:执行引擎遇到了任意一个方法返回的字节码指令或遇到了异常,并且该异常没有在方法体内得到处理。无论采用何种退出方式,在方法退出之后,都需要返回到方法被调用的位置,程序才能继续执行。方法返回时可能需要在栈帧中保存一些信息,用来帮助恢复它的上层方法的执行状态。一般来说,方法正常退出时,调用者的PC计数器的值就可以作为返回地址,栈帧中很可能保存了这个计数器值,而方法异常退出时,返回地址是要通过异常处理器来确定的,栈帧中一般不会保存这部分信息。
    方法退出的过程实际上等同于把当前栈帧出栈,因此退出时可能执行的操作有:恢复上层方法的局部变量表和操作数栈,如果有返回值,则把它压入调用者栈帧的操作数栈中,调整PC计数器的值以指向方法调用指令后面的一条指令。

  1. 本地方法栈
    本地方法栈也是线程私有的内存区域,与java栈比较相似,不同之处在于该区域主要是保存Native方法相关的数据。Native方法是非Java语言编写的方法。

    与虚拟机栈一样,本地方法栈区域也会抛出StackOverflowError和OutOfMemoryError异常。

  2. 程序计数器
    程序计数器是用于标识当前线程执行的字节码文件的行号指示器。多线程情况下,每个线程都具有各自独立的程序计数器,所以该区域是非线程共享的内存区域。

    当执行java方法时候,计数器中保存的是字节码文件的行号;当执行Native方法时,计数器的值为空。

三、JVM内存参数设置

在这里插入图片描述

  • -Xms设置堆的最小空间大小,默认是物理内存的1/64。
  • -Xmx设置堆的最大空间大小,默认是物理内存的1/4。
  • -XX:NewSize设置新生代最小空间大小。
  • -XX:MaxNewSize设置新生代最大空间大小。
  • -XX:PermSize设置永久代最小空间大小。
  • -XX:MaxPermSize设置永久代最大空间大小。
  • -Xss设置每个线程的堆栈大小。
JVM(Java Virtual Machine)即Java虚拟机,是一种用于计算设备的规范,它是一个虚构出来的计算机,通过在实际的计算机上仿真模拟各种计算机功能来实现 [^1]。JVM内存模型主要包含以下几个部分: ### 程序计数器 程序计数器用于记录当前执行的指令地址。如果程序执行的是一个Java方法,计数器记录的是正在执行的虚拟机字节码指令地址;若正在执行的是一个本地(native,由C语言编写完成)方法,则计数器的值为Undefined。由于它只是记录当前指令地址,不存在内存溢出的情况,是所有JVM内存区域中唯一一个没有定义OutOfMemoryError的区域 [^5]。 ### Java堆 Java堆是Java代码可及的内存,是留给开发人员使用的。堆用于存储对象实例及数组值,Java中所有通过new创建的对象的内存都在此分配,堆区由所有线程共享。Heap中对象所占用的内存由GC进行回收。在32位操作系统上最大为2GB,在64位操作系统上则没有限制,其大小可通过 -Xms 和 -Xmx 来控制。-Xms 为JVM启动时申请的最小Heap内存,默认为物理内存的1/64但小于1GB;-Xmx 为JVM可申请的最大Heap内存,默认为物理内存的1/4但小于1GB。默认当空余堆内存小于40%时,JVM会增大Heap到 -Xmx 指定的大小,可通过 -XX:MinHeapFreeRatio= 来指定这个比例;当空余堆内存大于70%时,JVM会减小Heap的大小到 -Xms 指定的大小,可通过 -XX:MaxHeapFreeRatio= 来指定这个比例。对于运行系统而言,为避免在运行时频繁调整Heap的大小,通常将 -Xms 和 -Xmx 的值设成一样 [^2][^3]。 ### 非堆(Non - Heap) 非堆是JVM留给自己用的,方法区、JVM内部处理或优化所需的内存 (如JIT编译后的代码缓存)、每个类结构 (如运行时常量池、字段和方法数据)以及方法和构造方法的代码都在非堆内存中 [^2]。 ### Java虚拟机栈 Java虚拟机栈是描述Java方法运行过程的内存模型。它会为每一个即将运行的Java方法创建“栈帧”,用于存储该方法在运行过程中所需要的一些信息,包括局部变量表(存放基本数据类型变量、引用类型的变量、returnAddress类型的变量)、操作数栈、动态链接、当前方法的常量池指针、当前方法的返回地址、方法出口等信息 [^4]。 ### 示例代码展示栈帧概念 ```java public class StackFrameExample { public static int add(int a, int b) { int result = a + b; return result; } public static void main(String[] args) { int x = 5; int y = 3; int sum = add(x, y); System.out.println("Sum: " + sum); } } ``` 在上述代码中,`main` 方法和 `add` 方法在执行时,Java虚拟机栈会分别为它们创建栈帧,用于存储各自运行过程中的信息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值