App下載

Java虛擬機(jī):工作原理、內(nèi)存管理、垃圾回收等

酷酷的小傻子 2023-07-02 11:00:00 瀏覽數(shù) (1357)
反饋

Java虛擬機(jī)(JVM)是一種基于棧式架構(gòu)的計(jì)算機(jī)程序,它可以將Java字節(jié)碼翻譯成特定的機(jī)器代碼。在這篇文章中,我們將深入探討JVM的工作原理、內(nèi)存管理和垃圾回收等方面,并結(jié)合具體實(shí)例進(jìn)行說明。

一、JVM的工作原理

JVM通過加載類文件并執(zhí)行其中的字節(jié)碼指令來運(yùn)行Java應(yīng)用程序。當(dāng)一個(gè)Java程序啟動(dòng)時(shí),JVM會(huì)首先加載所需的類文件,并將這些類文件轉(zhuǎn)換為一組可以被JVM處理的數(shù)據(jù)結(jié)構(gòu)。這些數(shù)據(jù)結(jié)構(gòu)包括方法區(qū)、堆、虛擬機(jī)棧、本地方法棧和程序計(jì)數(shù)器等。

  1. 方法區(qū)

方法區(qū)是JVM中的一塊內(nèi)存空間,用于存儲(chǔ)已加載的類信息、常量池、靜態(tài)變量等數(shù)據(jù)。例如:

Copy Code
public class Test { public static final String CONSTANT = "Hello, world!"; }

在上述代碼中,Test類的常量“Hello, world!”就存放在方法區(qū)的常量池中。

   2. 堆

堆是JVM中的另一塊內(nèi)存空間,用于存儲(chǔ)對(duì)象實(shí)例。當(dāng)程序創(chuàng)建一個(gè)對(duì)象時(shí),JVM會(huì)在堆中為該對(duì)象分配內(nèi)存空間,并返回一個(gè)指向該對(duì)象的引用。例如:

public class Test {
public void createObject() { Object obj = new Object(); } }

在上述代碼中,createObject方法會(huì)在堆中創(chuàng)建一個(gè)新的Object實(shí)例,并將其賦值給obj變量。

   3. 虛擬機(jī)棧

虛擬機(jī)棧是JVM中的一塊內(nèi)存空間,用于存儲(chǔ)方法調(diào)用時(shí)的局部變量、操作數(shù)棧、返回值等數(shù)據(jù)。每當(dāng)程序執(zhí)行一個(gè)方法時(shí),JVM就會(huì)為該方法創(chuàng)建一個(gè)新的棧幀,并將其壓入虛擬機(jī)棧中。例如:

public class Test {
public int add(int a, int b) { return a + b; } }

在上述代碼中,add方法會(huì)在虛擬機(jī)棧中創(chuàng)建一個(gè)新的棧幀,并將a和b的值分別存放在棧幀的局部變量表中。

   4. 本地方法棧

本地方法棧與虛擬機(jī)棧類似,但它是用于執(zhí)行本地方法(即由非Java語言編寫的方法)的??臻g。

   5. 程序計(jì)數(shù)器

程序計(jì)數(shù)器是JVM中的另一塊內(nèi)存空間,用于記錄當(dāng)前線程正在執(zhí)行的字節(jié)碼指令地址。每當(dāng)一個(gè)線程開始執(zhí)行一個(gè)方法時(shí),JVM就會(huì)將該方法的字節(jié)碼指令地址存放在程序計(jì)數(shù)器中,并且在執(zhí)行過程中不斷地更新程序計(jì)數(shù)器的值。

二、JVM的內(nèi)存管理

JVM的內(nèi)存管理主要包括堆內(nèi)存和方法區(qū)內(nèi)存的管理。

   1. 堆內(nèi)存管理

堆內(nèi)存由新生代和老年代兩部分組成。在新生代中,又分為Eden區(qū)、Survivor區(qū)0和Survivor區(qū)1三個(gè)區(qū)域。當(dāng)一個(gè)Java程序創(chuàng)建一個(gè)對(duì)象時(shí),JVM會(huì)在Eden區(qū)中為該對(duì)象分配內(nèi)存空間,并將其標(biāo)記為“Young Generation”(即屬于新生代)。當(dāng)Eden區(qū)滿了之后,JVM會(huì)觸發(fā)一次Minor GC(即新生代垃圾回收),將所有不再使用的對(duì)象從新生代中清除掉,并將還存活著的對(duì)象移動(dòng)到Survivor區(qū)0或Survivor區(qū)1中。

當(dāng)Survivor區(qū)0或Survivor區(qū)1也滿了之后,JVM會(huì)觸發(fā)一次Minor GC,將Survivor區(qū)中的所有不再使用的對(duì)象清除掉,并將還存活著的對(duì)象移動(dòng)到另一個(gè)空閑的Survivor區(qū)中。這樣,經(jīng)過多次Minor GC后,仍然存活的對(duì)象就會(huì)被移動(dòng)到老年代中。

在老年代中,由于對(duì)象生命周期較長(zhǎng),因此垃圾回收的頻率也較低。當(dāng)老年代空間不足時(shí),JVM會(huì)觸發(fā)一次Major GC(即Full GC),對(duì)整個(gè)堆內(nèi)存進(jìn)行垃圾回收。

   2. 方法區(qū)內(nèi)存管理

方法區(qū)內(nèi)存主要用于存儲(chǔ)已加載的類信息、常量池、靜態(tài)變量等數(shù)據(jù)。隨著應(yīng)用程序的運(yùn)行,方法區(qū)中可能會(huì)出現(xiàn)大量無用的類信息、常量和靜態(tài)變量,這些數(shù)據(jù)會(huì)占用大量的內(nèi)存空間。為了避免方法區(qū)內(nèi)存溢出,JVM會(huì)對(duì)方法區(qū)進(jìn)行垃圾回收。

但是,與堆內(nèi)存不同的是,方法區(qū)中的垃圾回收主要是針對(duì)常量池和類的卸載。如果一個(gè)類已經(jīng)被加載到方法區(qū)中,那么它就不能被卸載。因此,在實(shí)際開發(fā)中,我們通常需要采取一些手段來避免出現(xiàn)類的泄漏或者無用的常量池,如使用弱引用或者軟引用等。

三、JVM的垃圾回收

JVM的垃圾回收主要通過標(biāo)記-清除算法和復(fù)制算法來實(shí)現(xiàn)。其中,新生代采用復(fù)制算法,老年代采用標(biāo)記-清除算法。

   1. 復(fù)制算法

復(fù)制算法將內(nèi)存空間分為兩塊,每次只使用其中一塊。當(dāng)這一塊內(nèi)存空間不足時(shí),JVM就會(huì)停止應(yīng)用程序的運(yùn)行,將所有存活的對(duì)象復(fù)制到另一塊空閑的內(nèi)存空間中,并清理原有的內(nèi)存空間。這樣做的好處是避免了內(nèi)存碎片的產(chǎn)生,但是也會(huì)消耗較多的內(nèi)存空間。

   2. 標(biāo)記-清除算法

標(biāo)記-清除算法是一種比較常見的垃圾回收算法。它將內(nèi)存空間分為已使用和未使用兩部分,首先標(biāo)記所有正在使用的內(nèi)存空間(即存活的對(duì)象),然后將未標(biāo)記的內(nèi)存空間(即垃圾對(duì)象)進(jìn)行清理。

標(biāo)記-清除算法的缺點(diǎn)是容易產(chǎn)生內(nèi)存碎片,從而影響垃圾回收的效率。因此,在實(shí)際應(yīng)用中,通常會(huì)采用更先進(jìn)的垃圾回收算法,如標(biāo)記-整理算法和分代回收等。

結(jié)論

JVM是Java應(yīng)用程序的核心,它通過將Java字節(jié)碼翻譯成特定的機(jī)器代碼來運(yùn)行Java應(yīng)用程序。在JVM中,內(nèi)存管理和垃圾回收是非常重要的一部分,它們直接影響著應(yīng)用程序的性能和穩(wěn)定性。因此,在開發(fā)Java應(yīng)用程序時(shí),我們必須充分了解JVM的工作原理、內(nèi)存管理和垃圾回收等方面,并針對(duì)實(shí)際情況進(jìn)行優(yōu)化和調(diào)整。


0 人點(diǎn)贊