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