Java虛擬機(jī)如何運(yùn)行Java字節(jié)碼
Java虛擬機(jī)如何運(yùn)行Java字節(jié)碼
以標(biāo)準(zhǔn) JDK 中的 HotSpot 虛擬機(jī)為例,從虛擬機(jī)以及底層硬件兩個(gè)角度來(lái)看Java 虛擬機(jī)具體是怎么運(yùn)行 Java 字節(jié)碼的。
虛擬機(jī)視角
從虛擬機(jī)視角來(lái)看,執(zhí)行 Java 代碼首先需要將它編譯而成的 class 文件加載到 Java 虛擬機(jī)中。
加載后的 Java 類(lèi)會(huì)被存放于方法區(qū)(Method Area)中。
實(shí)際運(yùn)行時(shí),虛擬機(jī)會(huì)執(zhí)行方法區(qū)內(nèi)的代碼。

在運(yùn)行過(guò)程中,每當(dāng)調(diào)用進(jìn)入一個(gè) Java 方法,Java 虛擬機(jī)會(huì)在當(dāng)前線程的 Java 方法棧中生成一個(gè)棧幀,用以存放局部變量以及字節(jié)碼的操作數(shù)。
這個(gè)棧幀的大小是提前計(jì)算好的,而且 Java 虛擬機(jī)不要求棧幀在內(nèi)存空間里連續(xù)分布。
當(dāng)退出當(dāng)前執(zhí)行的方法時(shí),不管是正常返回還是異常返回,Java 虛擬機(jī)均會(huì)彈出當(dāng)前線程的當(dāng)前棧幀,并將之舍棄。
硬件視角
從硬件視角來(lái)看,Java 字節(jié)碼無(wú)法直接執(zhí)行。因此,Java 虛擬機(jī)需要將字節(jié)碼翻譯成機(jī)器碼。
在 HotSpot 里面,上述翻譯過(guò)程有兩種形式:
- 第一種是解釋執(zhí)行,即逐條將字節(jié)碼翻譯成機(jī)器碼并執(zhí)行;
- 第二種是即時(shí)編譯(Just-In-Time compilation,JIT),即將一個(gè)方法中包含的所有字節(jié)碼編譯成機(jī)器碼后再執(zhí)行。
前者的優(yōu)勢(shì)在于無(wú)需等待編譯,而后者的優(yōu)勢(shì)在于實(shí)際運(yùn)行速度更快。

HotSpot 默認(rèn)采用混合模式,綜合了解釋執(zhí)行和即時(shí)編譯兩者的優(yōu)點(diǎn)。
它會(huì)先解釋執(zhí)行字節(jié)碼,而后將其中反復(fù)執(zhí)行的熱點(diǎn)代碼,以方法為單位進(jìn)行即時(shí)編譯。
即時(shí)編譯
即時(shí)編譯建立在程序符合二八定律的假設(shè)上,也就是百分之二十的代碼占據(jù)了百分之八十的計(jì)算資源。
對(duì)于占據(jù)大部分的不常用的代碼,我們無(wú)需耗費(fèi)時(shí)間將其編譯成機(jī)器碼,而是采取解釋執(zhí)行的方式運(yùn)行;
另一方面,對(duì)于僅占據(jù)小部分的熱點(diǎn)代碼,我們則可以將其編譯成機(jī)器碼,以達(dá)到理想的運(yùn)行速度。
為了滿(mǎn)足不同用戶(hù)場(chǎng)景的需要,HotSpot 內(nèi)置了多個(gè)即時(shí)編譯器:C1、C2 和 Graal。
Graal 是 Java 10 正式引入的實(shí)驗(yàn)性即時(shí)編譯器,這里暫不做討論。
之所以引入多個(gè)即時(shí)編譯器,是為了在編譯時(shí)間和生成代碼的執(zhí)行效率之間進(jìn)行取舍。
- C1 又叫做 Client 編譯器,面向的是對(duì)啟動(dòng)性能有要求的客戶(hù)端 GUI 程序,采用的優(yōu)化手段相對(duì)簡(jiǎn)單,因此編譯時(shí)間較短。
- C2 又叫做 Server 編譯器,面向的是對(duì)峰值性能有要求的服務(wù)器端程序,采用的優(yōu)化手段相對(duì)復(fù)雜。
因此編譯時(shí)間較長(zhǎng),但同時(shí)生成代碼的執(zhí)行效率較高。
從 Java 7 開(kāi)始,HotSpot 默認(rèn)采用分層編譯的方式:熱點(diǎn)方法首先會(huì)被 C1 編譯,而后熱點(diǎn)方法中的熱點(diǎn)會(huì)進(jìn)一步被 C2 編譯。
為了不干擾應(yīng)用的正常運(yùn)行,HotSpot 的即時(shí)編譯是放在額外的編譯線程中進(jìn)行的。
HotSpot 會(huì)根據(jù) CPU 的數(shù)量設(shè)置編譯線程的數(shù)目,并且按 1:2 的比例配置給 C1 及 C2 編譯器。
在計(jì)算資源充足的情況下,字節(jié)碼的解釋執(zhí)行和即時(shí)編譯可同時(shí)進(jìn)行。
編譯完成后的機(jī)器碼會(huì)在下次調(diào)用該方法時(shí)啟用,以替換原本的解釋執(zhí)行。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java對(duì)象與json對(duì)象間的相互轉(zhuǎn)換的方法
本篇文章主要介紹了java對(duì)象與json對(duì)象間的相互轉(zhuǎn)換的方法,詳細(xì)介紹了json字符串和java對(duì)象相互轉(zhuǎn)換,有興趣的可以了解一下2017-01-01
詳解使用Spring3 實(shí)現(xiàn)用戶(hù)登錄以及權(quán)限認(rèn)證
這篇文章主要介紹了詳解使用Spring3 實(shí)現(xiàn)用戶(hù)登錄以及權(quán)限認(rèn)證,這里整理了詳細(xì)的代碼,有需要的小伙伴可以參考下。2017-03-03
解決mybatis-generator生成Mapper文件沒(méi)有Selective結(jié)尾的問(wèn)題
這篇文章主要介紹了解決mybatis-generator生成Mapper文件沒(méi)有Selective結(jié)尾的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09
SpringBoot中properties,yml,yaml的區(qū)別及使用說(shuō)明
這篇文章主要介紹了SpringBoot中properties,yml,yaml的區(qū)別及使用說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2025-03-03
elasticsearch數(shù)據(jù)信息索引操作action?support示例分析
這篇文章主要為大家介紹了elasticsearch數(shù)據(jù)信息索引操作action?support示例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-04-04
IDEA導(dǎo)入geoserver項(xiàng)目的詳細(xì)步驟及注意事項(xiàng)
由于GeoServer是基于Java開(kāi)發(fā)的。因此在安裝之前,必須確保安裝了Java。本文給大家分享IDEA導(dǎo)入geoserver項(xiàng)目的詳細(xì)步驟及注意事項(xiàng),感興趣的朋友一起看看吧2021-06-06
IntelliJ?IDEA?代碼運(yùn)行時(shí)中文出現(xiàn)亂碼問(wèn)題及解決方法
在我們剛接觸到IDEA時(shí),想美滋滋的敲一個(gè)“hello?world”來(lái)問(wèn)候這個(gè)世界,但難免會(huì)遇到這種問(wèn)題亂碼,這篇文章主要介紹了解決IntelliJ?IDEA?代碼運(yùn)行時(shí)中文出現(xiàn)亂碼問(wèn)題,需要的朋友可以參考下2023-09-09

