Java 內(nèi)存安全問(wèn)題的注意事項(xiàng)
前言
Java在內(nèi)存管理方面是要比C/C++更方便的,不需要為每一個(gè)對(duì)象編寫釋放內(nèi)存的代碼,JVM虛擬機(jī)將為我們選擇合適的時(shí)間釋放內(nèi)存空間,使得程序不容易出現(xiàn)內(nèi)存泄漏和溢出的問(wèn)題
不過(guò),也正是因?yàn)镴ava把內(nèi)存控制的權(quán)利交給了Java虛擬機(jī),一旦出現(xiàn)內(nèi)存泄漏和溢出方面的問(wèn)題,如果不了解虛擬機(jī)是怎么使用內(nèi)存的,那排查錯(cuò)誤將會(huì)成為一項(xiàng)異常艱難的工作
下面先看看JVM如何管理內(nèi)存的
內(nèi)存管理
根據(jù)Java虛擬機(jī)規(guī)范(第3版) 的規(guī)定,Java虛擬機(jī)所管理的內(nèi)存將會(huì)包括以下幾個(gè)運(yùn)行內(nèi)存數(shù)據(jù)區(qū)域:
- 線程隔離數(shù)據(jù)區(qū):
- 程序計(jì)數(shù)器: 當(dāng)前線程所執(zhí)行字節(jié)碼的行號(hào)指示器
- 虛擬機(jī)棧: 里面的元素叫棧幀,存儲(chǔ)局部變量表、操作棧、動(dòng)態(tài)鏈接、方法出口等,方法被調(diào)用到執(zhí)行完成的過(guò)程對(duì)應(yīng)一個(gè)棧幀在虛擬機(jī)棧中入棧到出棧的過(guò)程。
- 本地方法棧: 和虛擬機(jī)棧的區(qū)別在于虛擬機(jī)棧為虛擬機(jī)執(zhí)行Java方法,本地方法棧為虛擬機(jī)使用到的本地Native方法服務(wù)。
- 線程共享數(shù)據(jù)區(qū):
- 方法區(qū): 可以描述為堆的一個(gè)邏輯部分,或者說(shuō)使用永久代來(lái)實(shí)現(xiàn)方法區(qū)。存儲(chǔ)已被虛擬機(jī)加載的類信息、常量、靜態(tài)變量、即時(shí)編譯器編譯后的代碼等數(shù)據(jù)。
- 堆: 唯一目的就是存放對(duì)象的實(shí)例,是垃圾回收管理器的主要區(qū)域,分為Eden、From/To Survivor空間。

Java各版本內(nèi)存管理改進(jìn)
下圖中永久代理解為堆的邏輯區(qū)域,移除永久代的工作從JDK7就已經(jīng)開(kāi)始了,部分永久代中的數(shù)據(jù)(常量池)在JDK7中就已經(jīng)轉(zhuǎn)移到了堆中,JDK8中直接去除了永久代,方法區(qū)中的數(shù)據(jù)大部分被移到堆里面,還剩下一些元數(shù)據(jù)被保存在元空間里

內(nèi)存溢出
- 內(nèi)存泄露Memory Leak: 申請(qǐng)的內(nèi)存空間沒(méi)有及時(shí)釋放,導(dǎo)致后續(xù)程序里這塊內(nèi)容永遠(yuǎn)被占用。
- 內(nèi)存溢出Out Of Memory: 要求的內(nèi)存超過(guò)了系統(tǒng)所能提供的
運(yùn)行時(shí)數(shù)據(jù)區(qū)域的常見(jiàn)異常
在JVM中,除了程序計(jì)數(shù)器外,虛擬機(jī)內(nèi)存的其他幾個(gè)運(yùn)行時(shí)數(shù)據(jù)區(qū)域都有發(fā)生OOM異常的可能。
堆內(nèi)存溢出
不斷的創(chuàng)建對(duì)象,并且保證GC Roots到對(duì)象之間有可達(dá)路徑來(lái)避免垃圾回收機(jī)制清除這些對(duì)象。
public class HeapOOM {
static class ObjectInHeap{
}
public static void main(String[] args) {
List<ObjectInHeap> list = new ArrayList();
while (true) {
list.add(new ObjectInHeap());
}
}
}
棧溢出
單個(gè)線程下不斷擴(kuò)大棧的深度引起棧溢出。
public class StackSOF {
private int stackLength = 1;
public void stackLeak() {
stackLength++;
stackLeak();
}
public static void main(String[] args) {
StackSOF sof = new StackSOF();
try {
sof.stackLeak();
} catch (Throwable e) {
System.out.println("Stack Length: " + sof.stackLength);
throw e;
}
}
}
循環(huán)的創(chuàng)建線程,達(dá)到最大棧容量。
public class StackOOM {
private void dontStop() {
while (true) {
}
}
public void stackLeadByThread() {
while (true) {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
dontStop();
}
});
thread.start();
}
}
public static void main(String[] args) {
StackOOM stackOOM = new StackOOM();
stackOOM.stackLeadByThread();
}
}
運(yùn)行時(shí)常量池溢出
不斷的在常量池中新建String,并且保持引用不釋放。
public class RuntimeConstantPoolOOM {
public static void main(String[] args) {
// 使用List保持著常量池的引用,避免Full GC回收常量池
List<String> list = new ArrayList<String>();
int i = 0;
while (true) {
// intern()方法使String放入常量池
list.add(String.valueOf(i++).intern());
}
}
}
方法區(qū)溢出
借助CGLib直接操作字節(jié)碼運(yùn)行時(shí)產(chǎn)生大量的動(dòng)態(tài)類,最終撐爆內(nèi)存導(dǎo)致方法區(qū)溢出。
public class MethodAreaOOM {
static class ObjectInMethod {
}
public static void main(final String[] args) {
// 借助CGLib實(shí)現(xiàn)
while (true) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(ObjectInMethod.class);
enhancer.setUseCache(false);
enhancer.setCallback(new MethodInterceptor() {
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
return methodProxy.invokeSuper(o, objects);
}
});
enhancer.create();
}
}
}
元空間溢出
助CG Lib運(yùn)行時(shí)產(chǎn)生大量動(dòng)態(tài)類,唯一的區(qū)別在于運(yùn)行環(huán)境修改為Java 1.8,設(shè)置-XX:MaxMetaspaceSize參數(shù),便可以收獲java.lang.OutOfMemoryError: Metaspace這一報(bào)錯(cuò)
本機(jī)直接內(nèi)存溢出
直接申請(qǐng)分配內(nèi)存(實(shí)際上并沒(méi)有真正向操作系統(tǒng)申請(qǐng)分配內(nèi)存,而是通過(guò)計(jì)算得知內(nèi)存無(wú)法分配,于是拋出異常)
public class DirectMemoryOOM {
private static final int _1MB = 1024 * 1024;
public static void main(String[] args) throws IllegalAccessException {
Field unsafeField = Unsafe.class.getDeclaredFields()[0];
unsafeField.setAccessible(true);
Unsafe unsafe = (Unsafe) unsafeField.get(null);
while (true) {
unsafe.allocateMemory(_1MB);
}
}
}
常見(jiàn)案例
在工作中一般會(huì)遇到有以下幾種情況導(dǎo)致內(nèi)存問(wèn)題
- 傳輸數(shù)據(jù)量過(guò)大
因?yàn)閭鬏敂?shù)量過(guò)大、或一些極端情況導(dǎo)致代碼中間結(jié)果對(duì)象數(shù)據(jù)量過(guò)大,過(guò)大的數(shù)據(jù)量撐爆內(nèi)存
- 查詢出大量對(duì)象
這個(gè)多為SQL語(yǔ)句設(shè)置問(wèn)題,SQL未設(shè)置分頁(yè),用戶一次查詢數(shù)據(jù)量過(guò)大、頻繁查詢SQL導(dǎo)致內(nèi)存堆積、或是未作判空處理導(dǎo)致WHERE條件為空查詢出超大數(shù)據(jù)量等
- 接口性能問(wèn)題導(dǎo)致
這類為外部接口性能較慢,占用內(nèi)存較大,并且短時(shí)間內(nèi)高QPS導(dǎo)致的,導(dǎo)致服務(wù)內(nèi)存不足,線程堆積或掛起進(jìn)而出現(xiàn)FullGC
- 元空間問(wèn)題
使用了大量的反射代碼,Java字節(jié)碼存取器生成的類不斷生成
問(wèn)題排查
使用jmap分析內(nèi)存泄漏
1.生成dump文件
jmap -dump:format=b,file=/xx/xx/xx.hprof pid
2.dump文件下載到本地
3.dump文件分析
可以使用MAT,MAT可作為Eclipse插件或一個(gè)獨(dú)立軟件使用,MAT是一個(gè)高性能、具備豐富功能的Java堆內(nèi)存分析工具,主要用來(lái)排查內(nèi)存泄漏和內(nèi)存浪費(fèi)的問(wèn)題。
使用MAT打開(kāi)上一部后綴名.hprof的dump文件

- Histogram:直方圖,各個(gè)類的實(shí)例,包括個(gè)數(shù)和大小,可以查看類引用和被引用的路徑。
- Dominator Tree:支配圖,列出所有線程和線程下面的那些對(duì)象占用的空間。
- Top Consumers:通過(guò)圖形列出消耗內(nèi)存多的實(shí)例。
- Leak Suspects:MAT自動(dòng)分析的內(nèi)存泄漏報(bào)表
可以用這個(gè)工具分析出什么對(duì)象什么線程占用內(nèi)存空間較大,對(duì)象是被什么引用的,線程內(nèi)有哪些資源占用很高
以運(yùn)行時(shí)常量池溢出為例
打開(kāi)Histogram類實(shí)例表
Objects是類的對(duì)象的數(shù)量;Shallow是對(duì)象本身占用內(nèi)存大小、不包含其他引用;
Retained是對(duì)象自己的Shallow加上直接或間接訪問(wèn)到對(duì)象的Shallow之和,也可以說(shuō)是GC之后可以回收的內(nèi)存總和
從圖中可以看出運(yùn)行時(shí)常量池溢出的情況,產(chǎn)生了大量的String和char[]實(shí)例

在char[]上右鍵可以得到上圖所有char[]對(duì)象的被引用路徑,可以看出這些char數(shù)組都是以String的形式存在ArrayList中,并且是由main這個(gè)線程運(yùn)行的
可以看出是main線程中新建了一個(gè)數(shù)組,其中存了32w+個(gè)長(zhǎng)度為6的char數(shù)組組成的String造成的內(nèi)存溢出

關(guān)于MAT的詳細(xì)使用可以從MAT官方教程學(xué)習(xí)更多
以上就是Java 內(nèi)存安全問(wèn)題的注意事項(xiàng)的詳細(xì)內(nèi)容,更多關(guān)于Java 內(nèi)存安全問(wèn)題的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- Java DWR內(nèi)存泄漏問(wèn)題解決方案
- macOS上使用gperftools定位Java內(nèi)存泄漏問(wèn)題及解決方案
- Java內(nèi)存模型可見(jiàn)性問(wèn)題相關(guān)解析
- Java內(nèi)存泄漏問(wèn)題處理方法經(jīng)驗(yàn)總結(jié)
- 解決Java導(dǎo)入excel大量數(shù)據(jù)出現(xiàn)內(nèi)存溢出的問(wèn)題
- 完美解決java讀取大文件內(nèi)存溢出的問(wèn)題
- 詳解Java中synchronized關(guān)鍵字的死鎖和內(nèi)存占用問(wèn)題
- 解析Java的JNI編程中的對(duì)象引用與內(nèi)存泄漏問(wèn)題
- JAVA程序內(nèi)存溢出問(wèn)題原因分析
- Java中典型的內(nèi)存泄露問(wèn)題和解決方法
- 基于Java 數(shù)組內(nèi)存分配的相關(guān)問(wèn)題
相關(guān)文章
SpringBoot與knife4j的整合使用過(guò)程
Knife4j?是一個(gè)基于Swagger構(gòu)建的開(kāi)源?JavaAPI文檔工具,主要包括兩大核心功能:文檔說(shuō)明和在線調(diào)試,這篇文章主要介紹了SpringBoot與knife4j的整合使用,需要的朋友可以參考下2024-08-08
全面理解java中的構(gòu)造方法以及this關(guān)鍵字的用法
本篇文章主要概述了如何用構(gòu)造方法初始化對(duì)象,this屬性名訪問(wèn)成員變量方法,和this()的用法,感興趣的小伙伴一起來(lái)學(xué)習(xí)吧2023-03-03
一個(gè)applicationContext 加載錯(cuò)誤導(dǎo)致的阻塞問(wèn)題及解決方法
這篇文章主要介紹了一個(gè)applicationContext 加載錯(cuò)誤導(dǎo)致的阻塞問(wèn)題及解決方法,需要的朋友可以參考下2018-11-11
Spring boot自定義http反饋狀態(tài)碼詳解
這篇文章主要給大家介紹了Spring boot自定義http反饋狀態(tài)碼的相關(guān)資料,文中介紹的非常詳細(xì),對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面跟著小編一起來(lái)學(xué)習(xí)學(xué)習(xí)吧。2017-06-06
Java實(shí)現(xiàn)解析第三方接口返回的json
在實(shí)際開(kāi)發(fā)過(guò)程中,免不了和其他公司進(jìn)行聯(lián)調(diào),調(diào)用第三方接口,這個(gè)時(shí)候我們就需要根據(jù)對(duì)方返回的數(shù)據(jù)進(jìn)行解析,獲得我們想要的字段,下面我們就來(lái)看看具體有哪些方法吧2024-01-01
深入了解Java中Synchronized關(guān)鍵字的實(shí)現(xiàn)原理
synchronized是JVM的內(nèi)置鎖,基于Monitor機(jī)制實(shí)現(xiàn),每一個(gè)對(duì)象都有一個(gè)與之關(guān)聯(lián)的監(jiān)視器?(Monitor),這個(gè)監(jiān)視器充當(dāng)了一種互斥鎖的角色,本文就詳細(xì)聊一聊Synchronized關(guān)鍵字的實(shí)現(xiàn)原理,需要的朋友可以參考下2023-06-06

