Java OOM 異常場(chǎng)景與排查過(guò)程(堆、棧、方法區(qū))
Java OOM 異常場(chǎng)景與排查(堆、棧、方法區(qū))
一、引言
在 Java 應(yīng)用程序開(kāi)發(fā)和運(yùn)行過(guò)程中,OutOfMemoryError(OOM)異常是一個(gè)常見(jiàn)且令人頭疼的問(wèn)題。
OOM 異常表示 Java 虛擬機(jī)(JVM)在嘗試分配更多內(nèi)存時(shí)無(wú)法滿足需求,這通常意味著程序的內(nèi)存使用出現(xiàn)了問(wèn)題。根據(jù)內(nèi)存區(qū)域的不同,OOM 異常主要可分為堆內(nèi)存溢出、棧內(nèi)存溢出和方法區(qū)內(nèi)存溢出。
下面我們將詳細(xì)探討這些異常場(chǎng)景以及相應(yīng)的排查方法。
二、堆內(nèi)存溢出(Heap Space)
2.1 異常場(chǎng)景
堆是 Java 虛擬機(jī)中用于存儲(chǔ)對(duì)象實(shí)例的區(qū)域,當(dāng)程序不斷創(chuàng)建新對(duì)象,且這些對(duì)象一直被引用而無(wú)法被垃圾回收時(shí),堆內(nèi)存會(huì)不斷被占用,最終導(dǎo)致堆內(nèi)存溢出。常見(jiàn)的情況包括:
- 內(nèi)存泄漏:對(duì)象已經(jīng)不再使用,但由于程序的設(shè)計(jì)問(wèn)題,這些對(duì)象仍然被引用,無(wú)法被垃圾回收。例如,在集合中添加了對(duì)象,但后續(xù)沒(méi)有正確移除不再使用的對(duì)象。
- 大對(duì)象創(chuàng)建:程序中創(chuàng)建了非常大的對(duì)象,如大數(shù)組、大集合等,超過(guò)了堆內(nèi)存的可用空間。
- 對(duì)象數(shù)量過(guò)多:程序在短時(shí)間內(nèi)創(chuàng)建了大量的對(duì)象,導(dǎo)致堆內(nèi)存無(wú)法容納。
2.2 示例代碼
import java.util.ArrayList; import java.util.List; public class HeapOOMExample { public static void main(String[] args) { List<byte[]> list = new ArrayList<>(); while (true) { list.add(new byte[1024 * 1024]); // 每次創(chuàng)建 1MB 的數(shù)組 } } }
2.3 排查方法
- 使用工具監(jiān)控堆內(nèi)存使用情況:可以使用 VisualVM、Java Mission Control 等工具來(lái)監(jiān)控堆內(nèi)存的使用情況,查看堆內(nèi)存的增長(zhǎng)趨勢(shì)、對(duì)象的分布等信息。
- 分析堆轉(zhuǎn)儲(chǔ)文件:在 JVM 啟動(dòng)時(shí)添加
-XX:+HeapDumpOnOutOfMemoryError
參數(shù),當(dāng)發(fā)生堆內(nèi)存溢出時(shí),JVM 會(huì)自動(dòng)生成堆轉(zhuǎn)儲(chǔ)文件(.hprof)。然后使用工具(如 Eclipse Memory Analyzer)來(lái)分析堆轉(zhuǎn)儲(chǔ)文件,找出占用大量?jī)?nèi)存的對(duì)象。
三、棧內(nèi)存溢出(Stack Overflow)
3.1 異常場(chǎng)景
Java 棧是用于存儲(chǔ)方法調(diào)用的局部變量、操作數(shù)棧、動(dòng)態(tài)鏈接等信息的區(qū)域。每個(gè)線程都有自己獨(dú)立的??臻g。棧內(nèi)存溢出通常是由于方法調(diào)用的深度過(guò)深,導(dǎo)致棧幀不斷入棧,最終耗盡??臻g。常見(jiàn)的情況包括:
- 遞歸調(diào)用沒(méi)有終止條件:遞歸方法在沒(méi)有正確的終止條件時(shí),會(huì)不斷地調(diào)用自身,導(dǎo)致棧幀無(wú)限增加。
- 方法調(diào)用鏈過(guò)長(zhǎng):程序中存在復(fù)雜的方法調(diào)用鏈,每個(gè)方法都會(huì)創(chuàng)建棧幀,當(dāng)調(diào)用鏈過(guò)長(zhǎng)時(shí),??臻g會(huì)被耗盡。
3.2 示例代碼
public class StackOverflowExample { public static void recursiveMethod() { recursiveMethod(); // 無(wú)限遞歸調(diào)用 } public static void main(String[] args) { recursiveMethod(); } }
3.3 排查方法
- 查看異常堆棧信息:棧內(nèi)存溢出時(shí),JVM 會(huì)拋出
StackOverflowError
異常,并輸出詳細(xì)的異常堆棧信息。通過(guò)分析堆棧信息,可以定位到出現(xiàn)問(wèn)題的方法。 - 減少遞歸深度或優(yōu)化方法調(diào)用鏈:檢查遞歸方法是否有正確的終止條件,或者考慮使用迭代的方式替代遞歸。對(duì)于復(fù)雜的方法調(diào)用鏈,可以進(jìn)行優(yōu)化,減少不必要的方法調(diào)用。
四、方法區(qū)內(nèi)存溢出(Metaspace)
4.1 異常場(chǎng)景
方法區(qū)主要用于存儲(chǔ)類的元數(shù)據(jù)信息,如類的定義、常量池、方法字節(jié)碼等。在 Java 8 及以后的版本中,方法區(qū)由元空間(Metaspace)實(shí)現(xiàn)。方法區(qū)內(nèi)存溢出通常是由于程序動(dòng)態(tài)生成大量的類,導(dǎo)致元空間無(wú)法容納這些類的元數(shù)據(jù)信息。常見(jiàn)的情況包括:
- 動(dòng)態(tài)代理頻繁使用:使用動(dòng)態(tài)代理技術(shù)會(huì)在運(yùn)行時(shí)生成新的代理類,當(dāng)頻繁使用動(dòng)態(tài)代理時(shí),會(huì)生成大量的代理類,占用元空間。
- 大量加載類:在一些框架(如 Spring、Hibernate 等)中,會(huì)動(dòng)態(tài)加載大量的類,如果沒(méi)有合理的類加載機(jī)制,會(huì)導(dǎo)致元空間內(nèi)存溢出。
4.2 示例代碼
import java.lang.reflect.Proxy; public class MetaspaceOOMExample { public static void main(String[] args) { while (true) { Proxy.newProxyInstance( MetaspaceOOMExample.class.getClassLoader(), new Class<?>[]{Runnable.class}, (proxy, method, args1) -> null ); } } }
4.3 排查方法
- 監(jiān)控元空間使用情況:使用 VisualVM 等工具監(jiān)控元空間的使用情況,查看元空間的增長(zhǎng)趨勢(shì)。
- 調(diào)整元空間大小:在 JVM 啟動(dòng)時(shí),可以通過(guò)
-XX:MetaspaceSize
和-XX:MaxMetaspaceSize
參數(shù)來(lái)調(diào)整元空間的初始大小和最大大小。 - 檢查類加載機(jī)制:確保程序中沒(méi)有不必要的類加載操作,避免動(dòng)態(tài)生成過(guò)多的類。
總結(jié)
Java OOM 異常是一個(gè)復(fù)雜的問(wèn)題,不同的內(nèi)存區(qū)域出現(xiàn) OOM 異常的原因和排查方法也有所不同。在開(kāi)發(fā)和運(yùn)維過(guò)程中,需要密切關(guān)注程序的內(nèi)存使用情況,合理調(diào)整 JVM 參數(shù),優(yōu)化代碼邏輯,以避免 OOM 異常的發(fā)生。當(dāng)出現(xiàn) OOM 異常時(shí),要根據(jù)異常的類型和具體情況,采用合適的排查方法,找出問(wèn)題的根源并進(jìn)行解決。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
MybatisPlus的IPage失效的問(wèn)題解決方案
這篇文章主要介紹了MybatisPlus的IPage失效的問(wèn)題解決方案,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01Java使用泛型Class實(shí)現(xiàn)消除模板代碼
Class作為實(shí)現(xiàn)反射功能的類,在開(kāi)發(fā)中經(jīng)常會(huì)用到,然而,當(dāng)Class遇上泛型后,事情就變得不是那么簡(jiǎn)單了,所以本文就來(lái)講講Java如何使用泛型Class實(shí)現(xiàn)消除模板代碼,需要的可以參考一下2023-06-06Java使用強(qiáng)大的Elastisearch搜索引擎實(shí)例代碼
本篇文章主要介紹了Java使用強(qiáng)大的Elastisearch搜索引擎實(shí)例代碼,具有一定的參考價(jià)值,有興趣的可以了解一下2017-05-05SpringMVC中的請(qǐng)求參數(shù)接收方式
這篇文章主要介紹了SpringMVC中的請(qǐng)求參數(shù)接收方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2025-04-04使用restTemplate遠(yuǎn)程調(diào)controller路徑取數(shù)據(jù)
這篇文章主要介紹了使用restTemplate遠(yuǎn)程調(diào)controller路徑取數(shù)據(jù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08Python連接Java Socket服務(wù)端的實(shí)現(xiàn)方法
這篇文章主要介紹了Python連接Java Socket服務(wù)端的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01SpringBoot?整合mybatis+mybatis-plus的詳細(xì)步驟
這篇文章主要介紹了SpringBoot?整合mybatis+mybatis-plus的步驟,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06