溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java程序是如何執(zhí)行的

發(fā)布時(shí)間:2020-12-14 14:47:35 來(lái)源:億速云 閱讀:325 作者:Leah 欄目:開(kāi)發(fā)技術(shù)

這篇文章將為大家詳細(xì)講解有關(guān)Java程序是如何執(zhí)行的,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對(duì)相關(guān)知識(shí)有一定的了解。

Java程序執(zhí)行過(guò)程

Java程序是如何執(zhí)行的

  • 步驟 1: 寫(xiě)源代碼,源代碼將以.java的文件格式保存在電腦硬盤中。

  • 步驟 2: 編譯器(compiler)檢查是否存在編譯期錯(cuò)誤(例如缺少分號(hào),關(guān)鍵字拼寫(xiě)錯(cuò)誤等)。若通過(guò)檢測(cè),編譯器就會(huì)將源代碼翻譯成字節(jié)碼(bytecode),以.class的文件格式保存在電腦硬盤中。

  • 步驟 3: 若要運(yùn)行此Java程序,JVM中會(huì)有一個(gè)叫類加載器(class loader)的內(nèi)置程序把字節(jié)碼從硬盤載入到正位于內(nèi)存中的JVM里去。

  • 步驟 4: JVM中還有一個(gè)叫字節(jié)碼校驗(yàn)器(bytecode verifier)的內(nèi)置程序檢測(cè)是否存在運(yùn)行期錯(cuò)誤(例如棧溢出)。若通過(guò)檢測(cè),字節(jié)碼校驗(yàn)器就會(huì)將字節(jié)碼傳遞給解釋器(interpreter)。

  • 步驟 5: 解釋器會(huì)對(duì)字節(jié)碼進(jìn)行逐行翻譯,將其翻譯成當(dāng)前所在系統(tǒng)可以理解的機(jī)器碼(machine code),

  • 步驟 6:將機(jī)器碼交給操作系統(tǒng),操作系統(tǒng)會(huì)以main方法作為入口開(kāi)始執(zhí)行程序。至此,一個(gè)Java程序就這樣運(yùn)行起來(lái)了。

細(xì)心的讀者可能注意到了,在流程圖中還涉及到一個(gè)叫JIT的東西在步驟中沒(méi)有被解釋。那么JIT編譯器(Just-In-Time Compiler)是如果參與進(jìn)程序的執(zhí)行過(guò)程中呢?讓我們來(lái)看以下兩個(gè)例子。

Java程序是如何執(zhí)行的

  • 情況 1: 解釋器對(duì)代碼進(jìn)行逐行解釋,正如我們?cè)诓襟E中所介紹的。

  • 情況 2: 這是JIT編譯器參與進(jìn)Java執(zhí)行過(guò)程的情況,JIT編譯器會(huì)掃描所有代碼并對(duì)其進(jìn)行優(yōu)化。例如此時(shí)它發(fā)現(xiàn)最后一行代碼是重復(fù)多余的,就會(huì)將其移除,只傳遞前4行代碼給解釋器。這樣解釋器就能運(yùn)行地更快速高效,畢竟少了一行多余的代碼需要翻譯。

當(dāng)然,這只是JIT編譯器的優(yōu)化手段之一,不同公司設(shè)計(jì)的JIT編譯器對(duì)Java程序的運(yùn)行會(huì)有不同的優(yōu)化方式。此外需要知道的是,JIT編譯器并不是每次都會(huì)參與到執(zhí)行過(guò)程中來(lái)。

內(nèi)存機(jī)制

在步驟3中我們談到字節(jié)碼會(huì)被類加載器載入到內(nèi)存,那么載入之后JVM是如何對(duì)其進(jìn)行內(nèi)存管理的呢?

通常,在載入內(nèi)存后,一個(gè)Java程序所占用的內(nèi)存會(huì)被大致分為3塊區(qū)域:堆(heap),棧(stack)和方法區(qū)(method area)。

Java程序是如何執(zhí)行的

堆:存放new出來(lái)的東西。

棧:存放局部變量。

方法區(qū):類型信息,字段信息,常量池(constant pool),靜態(tài)變量,方法信息等。

public final class Student extends Object implements Serializable
{ 
// 1.類信息 
// 2.對(duì)象字段信息 
private String name; 
private int score; 
// 3.常量池 
public final int id = 0; 
public final String gender = "male"; 
// 4.靜態(tài)變量 
public static int a = 0; 
// 5.方法信息 
public int getid() 
{ 
return id; 
}}

PC寄存器:存放將要執(zhí)行的指令的地址。(因?yàn)闄C(jī)器的腦子不靈活,所以需要一塊專門的區(qū)域幫他記住執(zhí)行到哪一步,不然它會(huì)忘記)

本地方法棧:與JVM棧所發(fā)揮的作用是非常相似的,其區(qū)別不過(guò)是JVM棧為Java方法服務(wù),而本地方法棧則是為使用到的Native方法服務(wù)。有的虛擬機(jī)(例如Sun HotSpot虛擬機(jī))甚至直接就把本地方法棧和虛擬機(jī)棧合二為一。

每個(gè)線程擁有各自獨(dú)立的(虛擬機(jī))棧、PC寄存器和本地方法棧。而堆和方法區(qū)則是所有線程共享的。

最后讓我們通過(guò)一個(gè)小例子來(lái)理解Java程序執(zhí)行時(shí)內(nèi)存的變化。

public class Person 
{
int id; int age; Person(int id1, int age1)
{ 
id = id1; age = age1; 
} 
public static void main(String[] args) 
{ 
Person Tom = new Person(1, 25); 
} 
}

首先,在stack中申請(qǐng)了一塊內(nèi)存,這塊內(nèi)存區(qū)域名字叫Tom,此時(shí)區(qū)域里存儲(chǔ)的內(nèi)容為null。

接著,調(diào)用Person的構(gòu)造方法,方法的參數(shù)屬于局部變量,因此在stack中有兩塊區(qū)域分別存放id1和age1。

通過(guò)構(gòu)造方法,可以new出來(lái)一個(gè)Person的對(duì)象,這個(gè)對(duì)象連帶著其成員變量會(huì)被存放在heap中。成員變量id和age的值由存放在stack中的局部變量id1和age1賦予。

最后,將這個(gè)對(duì)象的引用值(類似于地址)傳遞給Tom,通過(guò)引用值我們就可以找到這個(gè)對(duì)象。

Java程序是如何執(zhí)行的

(注意:位于stack中的id1和age1會(huì)隨著構(gòu)造方法調(diào)用的結(jié)束而消失,這里為了更好地表現(xiàn)全過(guò)程,因此保留在圖中。)

關(guān)于棧和堆的其他小知識(shí)

  • 棧和堆的大小都是固定為一個(gè)默認(rèn)值的,它們作為jvm的參數(shù)設(shè)定好了,不同的jvm設(shè)定的參數(shù)不同,相應(yīng)的棧和堆的大小也就不同。

  • 棧是運(yùn)行時(shí)的單位:里面存儲(chǔ)的信息都是跟當(dāng)前線程相關(guān)的,包括局部變量、程序運(yùn)行狀態(tài)、方法返回值等;而堆是存儲(chǔ)的單位:它只負(fù)責(zé)存儲(chǔ)對(duì)象。

  • 當(dāng)一個(gè)方法調(diào)用結(jié)束后,方法里的局部變量會(huì)隨之消失,不會(huì)在stack中繼續(xù)占用空間。棧與堆的分離使得不同線程可以訪問(wèn)同一個(gè)對(duì)象,這是一種有效的數(shù)據(jù)交互方式(共享內(nèi)存);此外也節(jié)省了空間,因?yàn)槎阎械墓蚕沓A亢途彺婵梢员凰袟TL問(wèn)。

關(guān)于Java程序是如何執(zhí)行的就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI