使用mtrace追蹤JVM堆外內(nèi)存泄露的方法
mtrace追蹤內(nèi)存泄露
glibc中提供了mtrace這個(gè)函數(shù)來開啟追蹤內(nèi)存分配的功能,開啟后每次應(yīng)用程序調(diào)用malloc或free函數(shù)時(shí),會(huì)將內(nèi)存分配釋放操作記錄在MALLOC_TRACE環(huán)境變量所指的文件里面,如下:
$ pid=`pgrep java` # 配置gdb不調(diào)試信號(hào),避免JVM收到信號(hào)后被gdb暫停 $ cat <<"EOF" > ~/.gdbinit handle all nostop noprint pass handle SIGINT stop print nopass EOF # 設(shè)置MALLOC_TRACE環(huán)境變量,將內(nèi)存分配操作記錄在malloc_trace.log里 $ gdb -q -batch -ex 'call setenv("MALLOC_TRACE", "./malloc_trace.log", 1)' -p $pid # 調(diào)用mtrace開啟內(nèi)存分配追蹤 $ gdb -q -batch -ex 'call mtrace()' -p $pid # 一段時(shí)間后,調(diào)用muntrace關(guān)閉追蹤 $ gdb -q -batch -ex 'call muntrace()' -p $pid
然后查看malloc_trace.log,內(nèi)容如下:
可以發(fā)現(xiàn),在開啟mtrace后,glibc將所有malloc、free操作都記錄了下來,通過從日志中找出哪些地方執(zhí)行了malloc后沒有free,即是內(nèi)存泄露點(diǎn)。
于是glibc又提供了一個(gè)mtrace命令,其作用就是找出上面說的執(zhí)行了malloc后沒有free的記錄,如下:
$ mtrace malloc_trace.log | less -n Memory not freed: ----------------- Address Size Caller 0x00007efe08008cc0 0x18 at 0x7efe726e8e5d 0x00007efe08008ea0 0x160 at 0x7efe726e8e5d 0x00007efe6cabca40 0x58 at 0x7efe715dc432 0x00007efe6caa9ad0 0x1bf8 at 0x7efe715e4b88 0x00007efe6caab6d0 0x1bf8 at 0x7efe715e4b88 0x00007efe6ca679c0 0x8000 at 0x7efe715e4947 # 按Caller分組統(tǒng)計(jì)一下,看看各Caller各泄露的次數(shù)及內(nèi)存量 $ mtrace malloc_trace.log | sed '1,/Caller/d'|awk '{s[$NF]+=strtonum($2);n[$NF]++;}END{for(k in s){print k,n[k],s[k]}}'|column -t 0x7efe715e4b88 1010 7231600 0x7efe715dc432 1010 88880 0x7efe715e4947 997 32669696 0x7efe726e8e5d 532 309800 0x7efe715eb2f4 1 72 0x7efe715eb491 1 38
可以發(fā)現(xiàn),0x7efe715e4b88這個(gè)調(diào)用點(diǎn),泄露了1010次,那怎么知道這個(gè)調(diào)用點(diǎn)在哪個(gè)函數(shù)里呢?
根據(jù)指令地址找函數(shù)
之前我們介紹過Linux進(jìn)程的虛擬內(nèi)存布局,如下:
- Stack:棧,向下擴(kuò)展,為線程分配的棧內(nèi)存。
- Memory Mapping Segment:內(nèi)存映射區(qū)域,通過mmap分配,如映射的*.so動(dòng)態(tài)庫、動(dòng)態(tài)分配的匿名內(nèi)存等。
- Heap:堆,向上擴(kuò)展,動(dòng)態(tài)分配內(nèi)存的區(qū)域。
- Data Segment:數(shù)據(jù)段,一般用來存儲(chǔ)如C語言中的全局變量。
- Code Segment:代碼段,對(duì)于JVM來說,它從bin/java二進(jìn)制文件加載而來。
而對(duì)于JVM來說,bin/java只是一個(gè)啟動(dòng)進(jìn)程的殼,真正的代碼基本都在動(dòng)態(tài)庫中,如libjvm.so、libzip.so等。
而在Linux中,動(dòng)態(tài)庫都是直接加載的,如下:
因此,通過如下步驟,即可知道某個(gè)指令地址來自哪個(gè)函數(shù),如下:
- 根據(jù)指令地址,找到其所屬的動(dòng)態(tài)庫,以及動(dòng)態(tài)庫在進(jìn)程虛擬內(nèi)存空間中的起始地址。
- 根據(jù)指令地址減去起始地址,算出指令在動(dòng)態(tài)庫中的偏移量地址。
- 反匯編動(dòng)態(tài)庫文件,根據(jù)偏移量地址查找指令所在函數(shù)。
- 找動(dòng)態(tài)庫及起始地址
$ pmap -x $pid -p -A 0x7efe715e4b88 Address Kbytes RSS Dirty Mode Mapping 00007efe715d9000 108 108 0 r-x-- /opt/jdk8u222-b10/jre/lib/amd64/libzip.so ---------------- ------- ------- ------- total kB 108 163232 160716
通過pmap的-A選項(xiàng),可以通過內(nèi)存地址找內(nèi)存映射區(qū)域,如上,Mapping列就是內(nèi)存映射區(qū)域?qū)?yīng)的動(dòng)態(tài)庫文件,而Address列是其在進(jìn)程虛擬內(nèi)存空間中的起始地址。
- 計(jì)算指令在動(dòng)態(tài)庫中的偏移量
# 指令地址減去動(dòng)態(tài)庫起始地址 $ printf "%x" $((0x7efe715e4b88-0x00007efe715d9000)) bb88
- 反匯編并查找指令
$ objdump -d /opt/jdk8u222-b10/jre/lib/amd64/libzip.so | less -n
可以發(fā)現(xiàn),進(jìn)程地址0x7efe715e4b88
上的指令,在inflateInit2_
函數(shù)中。
當(dāng)然,上面步驟有點(diǎn)復(fù)雜,其實(shí)也可以通過gdb來查,如下:
gdb -q -batch -ex 'info symbol 0x7efe715e4b88' -p $pid
這樣,我們找到了泄露的原生函數(shù)名,那是什么java代碼調(diào)用到這個(gè)函數(shù)的呢?
通過原生函數(shù)名找Java調(diào)用棧
通過arthas的profiler命令,可以采樣到原生函數(shù)的調(diào)用棧,如下:
[arthas@1]$ profiler execute 'start,event=inflateInit2_,alluser' Profiling started [arthas@1]$ profiler stop OK profiler output file: .../arthas-output/20230923-173944.html
打開這個(gè)html文件,可以發(fā)現(xiàn)相關(guān)的Java調(diào)用棧,如下:
外內(nèi)存泄露的代碼路徑就找到了,只需要再看看代碼,識(shí)別一下哪些代碼路徑確實(shí)會(huì)導(dǎo)致內(nèi)存泄露即可。
注:經(jīng)過測試,發(fā)現(xiàn)profiler其實(shí)可以直接使用指令地址,所以不轉(zhuǎn)換為函數(shù)名稱,也是OK的。
通過jna開啟mtrace
gdb實(shí)際是C/C++的調(diào)試程序,通過gdb來直接調(diào)用native函數(shù),可能會(huì)出現(xiàn)一些不確定因素。
眾所周知,Java提供了JNI機(jī)制,可實(shí)現(xiàn)Java調(diào)用native函數(shù),而jna(Java Native Access)則對(duì)JNI技術(shù)進(jìn)行了封裝,大大簡化了Java調(diào)用native函數(shù)的開發(fā)工作。
因此,我們可以使用jna來調(diào)用mtrace等native函數(shù),如下:
- 引入jna庫
<dependency> <groupId>net.java.dev.jna</groupId> <artifactId>jna</artifactId> <version>4.2.2</version> </dependency>
- 封裝并調(diào)用native函數(shù)
public class JnaTool { public interface CLibrary extends Library { void malloc_stats(); void malloc_trim(int pad); void setenv(String name, String value, int overwrite); void mtrace(); void muntrace(); } private static CLibrary cLibrary; static { try { cLibrary = (CLibrary) Native.loadLibrary("c", CLibrary.class); } catch (Exception e) { e.printStackTrace(); } } public static void mtrace(String traceFile) { if (cLibrary == null) return; cLibrary.setenv("MALLOC_TRACE", traceFile, 1); cLibrary.mtrace(); } public static void muntrace() { if (cLibrary == null) return; cLibrary.muntrace(); } public static void mallocStats() { if (cLibrary == null) return; cLibrary.malloc_stats(); } public static void mallocTrim() { if (cLibrary == null) return; cLibrary.malloc_trim(0); } }
這樣,就可以避免使用gdb而調(diào)用一些C庫函數(shù)了
以上就是使用mtrace追蹤JVM堆外內(nèi)存泄露的方法的詳細(xì)內(nèi)容,更多關(guān)于mtrace追蹤JVM內(nèi)存泄露的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
java 靜態(tài)工廠代替多參構(gòu)造器的適用情況與優(yōu)劣
這篇文章主要介紹了java 靜態(tài)工廠代替多參構(gòu)造器的優(yōu)劣,幫助大家更好的理解和使用靜態(tài)工廠方法,感興趣的朋友可以了解下2020-12-12java顯示當(dāng)前運(yùn)行時(shí)的參數(shù)(java運(yùn)行參數(shù))
這篇文章主要介紹了java顯示當(dāng)前運(yùn)行時(shí)參數(shù)的示例(java運(yùn)行參數(shù)),需要的朋友可以參考下2014-04-04spring boot 實(shí)現(xiàn)配置多個(gè)DispatcherServlet最簡單方式
這篇文章主要介紹了spring boot 實(shí)現(xiàn)配置多個(gè)DispatcherServlet最簡單方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2021-01-01使用redis的increment()方法實(shí)現(xiàn)計(jì)數(shù)器功能案例
這篇文章主要介紹了使用redis的increment()方法實(shí)現(xiàn)計(jì)數(shù)器功能案例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-11-11SpringBoot使用@Validated處理校驗(yàn)的方法步驟
@Validated?注解的主要目的是啟用和利用?Spring?的驗(yàn)證框架,它可以用于類上也可以用于方法參數(shù)上,本文給大家介紹了SpringBoot使用@Validated優(yōu)雅的處理校驗(yàn)的方法步驟,通過代碼示例介紹的非常詳細(xì),需要的朋友可以參考下2024-08-08IDEA 如何導(dǎo)入別人的javaweb項(xiàng)目進(jìn)行部署
這篇文章主要介紹了IDEA 如何導(dǎo)入別人的javaweb項(xiàng)目進(jìn)行部署,本文給大家分享我的詳細(xì)部署過程及遇到問題解決方法,需要的朋友可以參考下2023-03-03