JVM: 内存、类与垃圾
Java虚拟机(JVM)是Java程序运行的核心,它负责执行Java字节码并管理Java应用程序的内存、类加载和垃圾回收机制。本文将深入探讨JVM的内存结构、类加载机制以及垃圾回收算法,帮助开发者更好地理解和优化Java应用。
一、JVM内存结构
JVM内存结构主要包括以下几个部分:
- 堆(Heap)
- 方法区(Method Area)
- 栈(Stack)
- 本地方法栈(Native Method Stack)
- 程序计数器(Program Counter Register)
1.1 堆(Heap)
堆是JVM中最大的一块内存区域,用于存储所有对象实例和数组。堆是垃圾回收的主要管理区域,根据对象的生命周期和存活时间进一步划分为新生代(Young Generation)和老年代(Old Generation)。
- 新生代:新创建的对象通常在新生代分配内存,分为Eden区和两个Survivor区(S0、S1)。
- 老年代:长期存活的对象会被移到老年代。
1.2 方法区(Method Area)
方法区用于存储类结构信息、常量、静态变量和即时编译后的代码。Java 8后,方法区由元空间(Metaspace)代替,元空间在本地内存中分配,而不是堆内存中。
1.3 栈(Stack)
栈用于存储每个线程的局部变量、操作数栈、动态链接和方法出口信息。每个方法调用都会在栈中创建一个栈帧。
1.4 本地方法栈(Native Method Stack)
本地方法栈与Java栈类似,但它为JVM执行本地(Native)方法服务,通常使用C语言实现。
1.5 程序计数器(Program Counter Register)
程序计数器是一个小内存区域,用于存储当前线程所执行的字节码的行号指示器。多线程切换时,程序计数器可以确保线程恢复到正确的执行位置。
二、类加载机制
JVM类加载机制分为以下几个步骤:
- 加载(Loading)
- 验证(Verification)
- 准备(Preparation)
- 解析(Resolution)
- 初始化(Initialization)
2.1 加载(Loading)
加载阶段通过类加载器(ClassLoader)读取类文件的二进制数据,并将其转换为方法区的数据结构。
2.2 验证(Verification)
验证阶段确保加载的类文件符合JVM规范,保证程序运行的安全性。
2.3 准备(Preparation)
准备阶段为类的静态变量分配内存,并将其初始化为默认值。
2.4 解析(Resolution)
解析阶段将常量池中的符号引用转换为直接引用。
2.5 初始化(Initialization)
初始化阶段根据程序员的意图对类的静态变量赋初值,并执行类的静态代码块。
三、垃圾回收机制
垃圾回收(Garbage Collection,GC)是JVM内存管理的重要机制,用于自动回收不再使用的对象所占用的内存。常见的垃圾回收算法包括:
- 标记-清除算法(Mark-Sweep)
- 复制算法(Copying)
- 标记-整理算法(Mark-Compact)
- 分代收集算法(Generational Collecting)
3.1 标记-清除算法(Mark-Sweep)
该算法分为“标记”和“清除”两个阶段。首先标记所有需要回收的对象,然后清除这些对象所占用的内存。
3.2 复制算法(Copying)
复制算法将内存划分为两块,每次只使用其中的一块。当一块内存用完时,将存活的对象复制到另一块空闲内存中,然后清除已用完的内存。
3.3 标记-整理算法(Mark-Compact)
该算法在标记阶段标记所有需要回收的对象,然后在整理阶段将存活的对象向内存的一端移动,最后清理掉边界外的内存。
3.4 分代收集算法(Generational Collecting)
分代收集算法将内存分为新生代和老年代,分别使用不同的垃圾回收算法。新生代对象使用复制算法,老年代对象使用标记-清除或标记-整理算法。