更新時(shí)間:2022-11-17 12:21:44 來(lái)源:動(dòng)力節(jié)點(diǎn) 瀏覽1474次
了解 Java 內(nèi)存模型是開(kāi)發(fā)、部署、監(jiān)視、測(cè)試和調(diào)優(yōu) Java 應(yīng)用程序性能的嚴(yán)肅 Java 開(kāi)發(fā)人員的基本學(xué)習(xí)內(nèi)容。在這篇博文中,我們將討論 Java 內(nèi)存模型以及 JVM 內(nèi)存模型的每個(gè)部分如何有助于運(yùn)行我們的程序。
首先看看你是否看懂下面這張JVM架構(gòu)圖。
在運(yùn)行資源密集型 Java 程序時(shí),您必須使用以下一些 JVM 內(nèi)存配置。
-XmsSetting — 初始堆大小
-XmxSetting — 最大堆大小
-XX:NewSizeSetting — 新一代堆大小
-XX:MaxNewSizeSetting — 最大新生代堆大小
-XX:MaxPermGenSetting — 永久代的最大大小
-XX:SurvivorRatioSetting — 新的堆大小比率(例如,如果 Young Gen 大小為 10m 且內(nèi)存開(kāi)關(guān)為 –XX:SurvivorRatio=2,則將為 Eden 空間保留 5m,為兩個(gè) Survivor 空間各保留 2.5m,默認(rèn)值 = 8)
-XX:NewRatio — 提供 Old/New Gen 大小的比率(默認(rèn)值 = 2)
但是你有沒(méi)有想過(guò)你的 JVM 是如何駐留在內(nèi)存中的?讓我展示一下。就像任何其他軟件一樣,JVM 會(huì)消耗主機(jī)操作系統(tǒng)內(nèi)存上的可用空間。
然而,在 JVM 內(nèi)部,存在單獨(dú)的內(nèi)存空間(Heap、Non-Heap、Cache),用于存儲(chǔ)運(yùn)行時(shí)數(shù)據(jù)和編譯后的代碼。
堆分為兩部分——年輕一代和老一輩
堆在 JVM 啟動(dòng)時(shí)分配(初始大小:-Xms)
堆大小在應(yīng)用程序運(yùn)行時(shí)增加/減少
最大尺寸:-Xmx
(1)年輕一代
這是為包含新分配的對(duì)象而保留的
Young Gen包括三個(gè)部分——Eden Memory和兩個(gè)Survivor Memory空間(S0,S1)
大多數(shù)新創(chuàng)建的對(duì)象進(jìn)入伊甸園空間。
當(dāng) Eden 空間充滿(mǎn)對(duì)象時(shí),將執(zhí)行Minor GC(又名Young Collection),并將所有幸存者對(duì)象移動(dòng)到其中一個(gè)幸存者空間。
Minor GC 還會(huì)檢查幸存者對(duì)象并將它們移動(dòng)到其他幸存者空間。所以在某個(gè)時(shí)候,其中一個(gè)幸存者空間總是空的。
經(jīng)過(guò)多次 GC 循環(huán)后幸存下來(lái)的對(duì)象將被移至老年代內(nèi)存空間。通常這是通過(guò)在年輕代對(duì)象有資格提升到老年代之前設(shè)置年齡閾值來(lái)完成的。
(2)老一代
這是保留的,用于包含在多輪 Minor GC 后可以存活的長(zhǎng)壽命對(duì)象
當(dāng) Old Gen 空間已滿(mǎn)時(shí),執(zhí)行Major GC(又名Old Collection)(通常需要更長(zhǎng)的時(shí)間)
這包括永久生成(自 Java 8 起由元空間取代)
Perm Gen 存儲(chǔ)每個(gè)類(lèi)的結(jié)構(gòu),例如運(yùn)行時(shí)常量池、字段和方法數(shù)據(jù)、方法和構(gòu)造函數(shù)的代碼,以及 interned 字符串
Its size can be changed using -XX:PermSize and -XX:MaxPermSize
這包括代碼緩存
存放JIT編譯器生成的編譯代碼(即native代碼)、JVM內(nèi)部結(jié)構(gòu)、加載的profiler agent代碼和數(shù)據(jù)等。
當(dāng)代碼緩存超過(guò)閾值時(shí),它會(huì)被刷新(對(duì)象不會(huì)被 GC 重新定位)。
到目前為止,我沒(méi)有提到任何關(guān)于 Java Stack 內(nèi)存的內(nèi)容,因?yàn)槲蚁雴为?dú)強(qiáng)調(diào)它的區(qū)別。首先,看看下面的圖片,看看你是否知道這里發(fā)生了什么。
總之,長(zhǎng)話(huà)短說(shuō),Java Stack 內(nèi)存用于線(xiàn)程的執(zhí)行,它包含方法特定的值和對(duì) Heap 中其他對(duì)象的引用。讓我們將 Stack 和 Heap 都放到一個(gè)表中,看看它們的區(qū)別。
這是一個(gè)很好的例子(來(lái)自 baeldung.com),它說(shuō)明了 Stack 和 Heap 如何有助于執(zhí)行一個(gè)簡(jiǎn)單的程序(使用代碼檢查堆棧順序)。
類(lèi)人 {
int pid;
字符串名稱(chēng);
// 構(gòu)造函數(shù),setter/getter
}
public class Driver {
public static void main(String[] args) {
int id = 23;
String pName = "喬恩";
人 p = null;
p = new Person(id, pName);
}
}
上面的 Java 內(nèi)存模型是最常討論的實(shí)現(xiàn)。然而,最新的 JVM 版本有不同的修改,例如引入了以下新的內(nèi)存空間。
Keep Area——新生代中的一個(gè)新內(nèi)存空間,用于包含最近分配的對(duì)象。直到下一個(gè)年輕一代才會(huì)執(zhí)行 GC。這個(gè)區(qū)域防止對(duì)象僅僅因?yàn)樗鼈兪窃谀贻p收集開(kāi)始之前分配的而被提升。
Metaspace——從 Java 8 開(kāi)始,Permanent Generation 被 Metaspace 取代。它可以自動(dòng)增加它的大小(直到底層操作系統(tǒng)提供的大小),即使 Perm Gen 總是有一個(gè)固定的最大大小。只要類(lèi)加載器處于活動(dòng)狀態(tài),元數(shù)據(jù)就會(huì)在元空間中保持活動(dòng)狀態(tài)并且不能被釋放。
注意: 始終建議您瀏覽供應(yīng)商文檔以找出適合您的 JVM 版本的內(nèi)容。
當(dāng)存在嚴(yán)重的內(nèi)存問(wèn)題時(shí),JVM 會(huì)崩潰并在您的程序輸出中拋出錯(cuò)誤指示,如下所示。
java.lang.StackOverFlowError — 表示Java堆棧內(nèi)存已滿(mǎn)
java.lang.OutOfMemoryError: Java heap space — 表示堆內(nèi)存已滿(mǎn)
java.lang.OutOfMemoryError: GC Overhead limit exceeded — 表示 GC 已達(dá)到其開(kāi)銷(xiāo)限制
java.lang.OutOfMemoryError: Permgen space — 表示永久代空間已滿(mǎn)
java.lang.OutOfMemoryError: Metaspace — 表示元空間已滿(mǎn)(自 Java 8 起)
java.lang.OutOfMemoryError: Unable to create new native thread — 表示 JVM 本機(jī)代碼無(wú)法再?gòu)牡讓硬僮飨到y(tǒng)創(chuàng)建新的本機(jī)線(xiàn)程,因?yàn)橐呀?jīng)創(chuàng)建了太多線(xiàn)程并且它們消耗了 JVM 的所有可用內(nèi)存
java.lang.OutOfMemoryError: request size bytes for reason — 表示交換內(nèi)存空間已被應(yīng)用程序完全消耗
java.lang.OutOfMemoryError: Requested array size exceeds VM limit – 表示我們的應(yīng)用程序使用的數(shù)組大小超過(guò)了底層平臺(tái)允許的大小
相關(guān)閱讀
0基礎(chǔ) 0學(xué)費(fèi) 15天面授
有基礎(chǔ) 直達(dá)就業(yè)
業(yè)余時(shí)間 高薪轉(zhuǎn)行
工作1~3年,加薪神器
工作3~5年,晉升架構(gòu)
提交申請(qǐng)后,顧問(wèn)老師會(huì)電話(huà)與您溝通安排學(xué)習(xí)