Java OOM 異常場(chǎng)景與排查過程(堆、棧、方法區(qū))
Java OOM 異常場(chǎng)景與排查(堆、棧、方法區(qū))
一、引言
在 Java 應(yīng)用程序開發(fā)和運(yùn)行過程中,OutOfMemoryError(OOM)異常是一個(gè)常見且令人頭疼的問題。
OOM 異常表示 Java 虛擬機(jī)(JVM)在嘗試分配更多內(nèi)存時(shí)無(wú)法滿足需求,這通常意味著程序的內(nèi)存使用出現(xià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)存溢出。常見的情況包括:
- 內(nèi)存泄漏:對(duì)象已經(jīng)不再使用,但由于程序的設(shè)計(jì)問題,這些對(duì)象仍然被引用,無(wú)法被垃圾回收。例如,在集合中添加了對(duì)象,但后續(xù)沒有正確移除不再使用的對(duì)象。
- 大對(duì)象創(chuàng)建:程序中創(chuàng)建了非常大的對(duì)象,如大數(shù)組、大集合等,超過了堆內(nèi)存的可用空間。
- 對(duì)象數(shù)量過多:程序在短時(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ú)立的棧空間。棧內(nèi)存溢出通常是由于方法調(diào)用的深度過深,導(dǎo)致棧幀不斷入棧,最終耗盡??臻g。常見的情況包括:
- 遞歸調(diào)用沒有終止條件:遞歸方法在沒有正確的終止條件時(shí),會(huì)不斷地調(diào)用自身,導(dǎo)致棧幀無(wú)限增加。
- 方法調(diào)用鏈過長(zhǎng):程序中存在復(fù)雜的方法調(diào)用鏈,每個(gè)方法都會(huì)創(chuàng)建棧幀,當(dāng)調(diào)用鏈過長(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ì)的異常堆棧信息。通過分析堆棧信息,可以定位到出現(xià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ù)信息。常見的情況包括:
- 動(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)加載大量的類,如果沒有合理的類加載機(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í),可以通過
-XX:MetaspaceSize和-XX:MaxMetaspaceSize參數(shù)來(lái)調(diào)整元空間的初始大小和最大大小。 - 檢查類加載機(jī)制:確保程序中沒有不必要的類加載操作,避免動(dòng)態(tài)生成過多的類。
總結(jié)
Java OOM 異常是一個(gè)復(fù)雜的問題,不同的內(nèi)存區(qū)域出現(xiàn) OOM 異常的原因和排查方法也有所不同。在開發(fā)和運(yùn)維過程中,需要密切關(guān)注程序的內(nèi)存使用情況,合理調(diào)整 JVM 參數(shù),優(yōu)化代碼邏輯,以避免 OOM 異常的發(fā)生。當(dāng)出現(xiàn) OOM 異常時(shí),要根據(jù)異常的類型和具體情況,采用合適的排查方法,找出問題的根源并進(jìn)行解決。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java使用泛型Class實(shí)現(xiàn)消除模板代碼
Class作為實(shí)現(xiàn)反射功能的類,在開發(fā)中經(jīng)常會(huì)用到,然而,當(dāng)Class遇上泛型后,事情就變得不是那么簡(jiǎn)單了,所以本文就來(lái)講講Java如何使用泛型Class實(shí)現(xiàn)消除模板代碼,需要的可以參考一下2023-06-06
Java使用強(qiáng)大的Elastisearch搜索引擎實(shí)例代碼
本篇文章主要介紹了Java使用強(qiáng)大的Elastisearch搜索引擎實(shí)例代碼,具有一定的參考價(jià)值,有興趣的可以了解一下2017-05-05
SpringMVC中的請(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-08
Python連接Java Socket服務(wù)端的實(shí)現(xiàn)方法
這篇文章主要介紹了Python連接Java Socket服務(wù)端的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01
SpringBoot?整合mybatis+mybatis-plus的詳細(xì)步驟
這篇文章主要介紹了SpringBoot?整合mybatis+mybatis-plus的步驟,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06

