JVM代碼運行邏輯解讀
JVM代碼運行邏輯
理解一個Java程序在JVM中的執(zhí)行流程,有助于深入理解JVM內(nèi)存管理和運行機制。
讓我們通過一個簡單的Java代碼示例,結(jié)合JVM的各個內(nèi)存區(qū)域和執(zhí)行過程,逐步分析一個Java程序在JVM中的執(zhí)行邏輯。
示例代碼
public class HelloWorld {
// 靜態(tài)變量存放在方法區(qū)
private static String greeting = "Hello, World!";
// 成員變量存放在堆內(nèi)存
private int number;
public HelloWorld(int number) {
this.number = number;
}
// main方法存放在方法區(qū),局部變量在棧上
public static void main(String[] args) {
// 局部變量存放在虛擬機棧
int localNumber = 10;
// 對象存儲在堆,引用變量存在棧中
HelloWorld hello = new HelloWorld(localNumber);
// 調(diào)用方法并傳遞引用
hello.printGreeting();
}
// 實例方法也存放在方法區(qū)
public void printGreeting() {
System.out.println(greeting + " Number: " + number);
}
}執(zhí)行過程分析
1.類加載
程序開始時,JVM 會通過類加載器(ClassLoader)加載 HelloWorld 類的字節(jié)碼文件(HelloWorld.class)到方法區(qū)中。
- 方法區(qū)(Java 8 以后叫元空間 Metaspace):存放類的信息,如類的字節(jié)碼、靜態(tài)變量、方法信息(
main方法、printGreeting方法等),以及常量池(如字符串常量池存儲"Hello, World!")。 - 靜態(tài)變量
greeting存放在方法區(qū),main方法的字節(jié)碼也會存儲在方法區(qū)。
2.JVM 啟動 main 方法
JVM 啟動后,它首先尋找 HelloWorld 類的 main 方法,執(zhí)行這個方法是程序的入口。此時會創(chuàng)建一個用于 main 方法的棧幀(Stack Frame),這個棧幀會存放 main 方法中的局部變量。
int localNumber = 10;
- 虛擬機棧(JVM Stack):
main方法的棧幀存放在虛擬機棧中。局部變量localNumber的值(10)也存儲在這個棧幀中。
3.對象分配
執(zhí)行 HelloWorld hello = new HelloWorld(localNumber); 時,JVM 會在堆內(nèi)存中分配一個新的 HelloWorld 對象,存儲這個對象的實例變量。
- 堆內(nèi)存(Heap):
new HelloWorld(10)會在堆中創(chuàng)建一個對象實例,并為number分配存儲空間。number的值(10)被初始化并存放在堆中。 hello變量本身是一個引用,它存儲在main方法的棧幀中,指向堆中的HelloWorld對象。
4.方法調(diào)用
當執(zhí)行 hello.printGreeting(); 時,JVM 會為 printGreeting 方法創(chuàng)建一個新的棧幀,并將 hello 作為隱含的 this 傳遞給該方法。
- 虛擬機棧(JVM Stack):
printGreeting方法的棧幀存儲局部變量和操作數(shù)棧,包括引用this(指向堆中的HelloWorld對象)。 - 在
printGreeting中,JVM 會從堆內(nèi)存中取出number的值(10),從方法區(qū)中的字符串常量池取出"Hello, World!",并執(zhí)行System.out.println。
5.輸出操作
System.out.println(greeting + " Number: " + number);
這行代碼通過 System.out 調(diào)用來輸出內(nèi)容到控制臺。
最終,控制臺會輸出:
Hello, World! Number: 10
- 方法區(qū):
greeting是一個靜態(tài)變量,它在類加載時就已經(jīng)存儲在方法區(qū)。"Hello, World!"存儲在字符串常量池中。 - 堆內(nèi)存:
this.number的值(10)從堆中獲取。
6.垃圾回收(GC)
當 main 方法執(zhí)行結(jié)束后,main 方法的棧幀被銷毀,局部變量 hello 的引用也會隨之消失。如果沒有其他地方再引用堆中的 HelloWorld 對象,那么垃圾回收器(GC)會認為它是不可達對象,從而在合適的時候?qū)υ搶ο筮M行回收,釋放它占用的堆內(nèi)存。
內(nèi)存區(qū)域的角色總結(jié)
1.方法區(qū)(元空間 Metaspace):
- 存儲類的元數(shù)據(jù)、靜態(tài)變量和常量池信息。
- 靜態(tài)變量
greeting和方法的字節(jié)碼存儲在這里。
2.堆內(nèi)存(Heap Memory):
- 存儲所有對象實例和數(shù)組。
HelloWorld對象實例存儲在堆內(nèi)存中,實例變量number也存在堆中。
3.虛擬機棧(JVM Stack):
- 每個線程都會有一個獨立的棧,用來存儲方法調(diào)用的局部變量和操作數(shù)棧。
main方法和printGreeting方法的棧幀存儲在虛擬機棧中,局部變量如localNumber和hello引用都在棧中。
4.程序計數(shù)器(Program Counter Register):
- 存儲當前線程正在執(zhí)行的字節(jié)碼指令地址。
- 線程切換時,程序計數(shù)器幫助恢復代碼的執(zhí)行位置。
5.本地方法棧(Native Method Stack):
- 專門用于執(zhí)行本地方法(如 JNI 調(diào)用的 C/C++ 代碼),但在此例子中未涉及。
總結(jié)
通過這段代碼,JVM 的執(zhí)行邏輯可以清晰地展現(xiàn)出來:
- 類加載器首先加載字節(jié)碼到方法區(qū)。
- 在堆內(nèi)存中分配對象,引用保存在虛擬機棧中。
- 方法的執(zhí)行流程通過棧幀的創(chuàng)建和銷毀來管理。
- 最終,垃圾回收器會在內(nèi)存中清理不再使用的對象。
這種分工明確的內(nèi)存管理機制使得 JVM 能夠高效運行,并保證內(nèi)存的安全性和自動管理。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Springboot中的@ConditionalOnBean注解詳細解讀
這篇文章主要介紹了Springboot中的@ConditionalOnBean注解詳細解讀,@ConditionalOnMissingBean注解兩個類,一個Computer類,一個配置類,想要完成;如果容器中沒有Computer類,就注入備用電腦Computer類,如果有Computer就不注入,需要的朋友可以參考下2023-11-11
springboot中@PostConstruct注解使用小結(jié)
本文主要介紹了springboot中@PostConstruct注解使用小結(jié),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2025-01-01
postman?如何實現(xiàn)傳遞?ArrayList?給后臺
這篇文章主要介紹了postman?如何實現(xiàn)傳遞?ArrayList給后臺,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-12-12
使用SpringMVC響應json格式返回的結(jié)果類型
這篇文章主要介紹了使用SpringMVC響應json格式返回的結(jié)果類型,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
詳解JAVA生成將圖片存入數(shù)據(jù)庫的sql語句實現(xiàn)方法
這篇文章主要介紹了詳解JAVA生成將圖片存入數(shù)據(jù)庫的sql語句實現(xiàn)方法的相關(guān)資料,這里就是實現(xiàn)java生成圖片并存入數(shù)據(jù)庫的實例,需要的朋友可以參考下2017-08-08
JAVA SpringBoot統(tǒng)一日志處理原理詳解
這篇文章主要介紹了SpringBoot的統(tǒng)一日志處理原理,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-09-09

