目录
- JVM代码运行逻辑
- 示例代码
- 执行过程分析
- 内存区域的角色总结
- 总结
JVM代码运行逻辑
理解一个Java程序在JVM中的执行流程,有助于深入理解JVM内存管理和运行机制。
让我们通过一个简单的Java代码示例,结合JVM的各个内存区域和执行过程,逐步分析一个Java程序在JVM中的执行逻辑。
示例代码
public class HelloWorld {
// 静态变量存放在方法区
private statPngKBmuKQic String greeting = "Hello, World!";
// 成员变量存放在堆内存
private int number;
public HelloWorld(int number) {
this.number = number;
}
// main方法存放在方法区,局部变量在栈上
public static void main(String[] args) {
// 局部变量存放在虚拟机栈
int localNumber = 10;
// 对象存储在堆,引用变量存在栈中
HelloWorld hello = new HelloWorld(localNumber);
// 调用方法并传递引用
hello.printGreeting();
}
// 实例方法也存放在方法区
public void printGreeting() {
System.out.println(greeting + " Number: " + number);
}
}
执行过程分析
1.类加载
程序开始时,JVM 会通过类加载器(ClassLoaPngKBmuKQder)加载 HelloWorld 类的字节码文件(HelloWorld.class)到方法区中。
- 方法区(Java 8 以后叫元空间 MetASPace):存放类的信息,如类的字节码、静态变量、方法信息(
main方法、printGreeting方法等),以及常量池(如字符串常量池存储"Hello, World!")。 - 静态变量
greeting存放在方法区,main方法的字节码也会存储在方法区。
2.JVM 启动 main 方法
JVM 启动后,它首先寻找 HelloWorld 类的 main 方法,执行这个方法是程序的入口。此时会创建一个用于 main 方法的栈帧(Stack Frame),这个栈帧会存放 main 方法中的局部变量。
int localNumber = 10;
- 虚拟机栈(JVM Stack):
main方法的栈帧存放在虚拟机栈中。局部变量localNumber的值(10)也存储在这个栈帧中。
3.对编程客栈象分配
执行 HelloWorld hello = new HelloWorld(localNumber); 时,JVM 会在堆内存中分配一个新的 HelloWorld 对象,存储这个对象的实例变量。
- 堆内存(Heap):
new HelloWorld(10)会在堆中创建一个对象实例,并为number分配存储空间。number的值(10)被初始化并存放在堆中。 hello变量本身是一个引用,它存储在main方法的栈帧中,指向堆中的HelloWorld对象。
4.方法调用
当执行 hello.printGreeting(); 时,JVM 会为 printGreeting 方法创建一个新的栈帧,并将 hello 作为隐含的 this 传递给该方法。
- 虚拟机栈(JVM Stack):
printGreeting方法的栈帧存储局部变量和操作数栈,包括引用this(指向堆中的HelloWorld对象)。 - 在
printGreeting中,JVM 会从堆内存中取出number的值(10),从方法区中的字符串常量池取出"Hello, World!",并执行System.out.println。
5.输出操作
System.out.println(greeting + " Number: " + number);
这行代码通过 System.out 调用来输出内容到控制台。
最终,控制台会输出:
Hello, World! Number: 10
- 方法区:
greeting是一个静态变量,它在类加载时就已经存储在方法区。"Hello, World!"存储在字符串常量池中。 - 堆内存:
this.number的值(10)从堆中获取。
6.垃圾回收(GC)
当 main 方法执行结束后,main 方法的栈帧被销毁,局部变量 hello 的引用也会随之消失。如果没有其他地方再引用堆中的 HelloWorld 对象,那么垃圾回收器(GC)会认为它是不可达对象,从而在合适的时候对该对象进行回收,释放它占用的堆内存。
内存区域的角色总结
1.方法区(元空间 Metaspace):
- 存储类的元数据、静态变量和常量池信息。
- 静态变量
greeting和方法的字节码存储在这里。
2.堆内存(Heap Memory):
- 存储所有对象实例和数组。
HelloWorld对象实例存储在堆内存http://www.devze.com中,实例变量number也存在堆中。
3.虚拟机栈(JVM Stack):
- 每个线程都会有一个独立的栈,用来存储方法调用的局部变量和操作数栈。
main方法和printGreeting方法的栈帧存储在虚拟机栈中,局部变量如localNumber和hello引用都在栈中。
4.程序计数器(Program Counter Register):
- 存储当前线程正在执行的字节码指令地址。
- 线程切换时,程序计数器帮助恢复代码的执行位置。
5.本地方法栈(Native Method Stack):
- 专门用于执行本地方法(如 JNI 调用的 C/C++ 代码),但在此例子中未涉及。
总结
通过这段代码,JVM 的执行逻辑可以清晰地展现出来:
- 类加载器首先加载字节码到方法区。
- 在堆内存中分配对象,引用保存在虚拟机栈中。
- 方法的执行流程通过栈帧php的创建和销毁来管理。
- 最终,垃圾回收器会在内存中清理不再使用的对象。
这种分工明确的内存管理机制使得 JVM 能够高效运行,并保证内存的安全性和自动管理。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程客栈(www.devze.com)。
加载中,请稍侯......
精彩评论