精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

如何使用NMT和pmap來解決JVM的資源泄漏問題

openEuler ? 來源:openEuler ? 作者:宋堯飛 ? 2021-09-24 16:00 ? 次閱讀

編者按:筆者使用 JDK 自帶的內(nèi)存跟蹤工具 NMT 和 Linux 自帶的 pmap 解決了一個非常典型的資源泄漏問題。這個資源泄漏是由于 Java 程序員不正確地使用 Java API 導(dǎo)致的,使用 Files.list 打開的文件描述符必須關(guān)閉。本案例一方面介紹了怎么使用 NMT 解決 JVM 資源泄漏問題,如果讀者遇到類似問題,可以嘗試用 NMT 來解決;另一方面也提醒 Java 開發(fā)人員使用 Java API 時需要必須弄清楚 API 使用規(guī)范,希望大家通過這個案例有所收獲。

背景知識:

NMT

NMT 是 Native Memory Tracking 的縮寫,一個 JDK 自帶的小工具,用來跟蹤 JVM 本地內(nèi)存分配情況(本地內(nèi)存指的是 non-heap,例如 JVM 在運(yùn)行時需要分配一些輔助數(shù)據(jù)結(jié)構(gòu)用于自身的運(yùn)行)。

NMT 功能默認(rèn)關(guān)閉,可以在 Java 程序啟動參數(shù)中加入以下參數(shù)來開啟:

-XX:NativeMemoryTracking=[summary | detail]

其中,“summary” 和 “detail” 的差別主要在輸出信息的詳細(xì)程度。

3cb43d90-10ac-11ec-8fb8-12bb97331649.png

開啟 NMT 功能后,就可以使用 JDK 提供的 jcmd 命令來讀取 NMT 采集的數(shù)據(jù)了,具體命令如下:

jcmd 《pid》 VM.native_memory [summary | detail | baseline | summary.diff | detail.diff | shutdown]

NMT 參數(shù)的含義可以通過 “jcmd 《pid》 help VM.native_memory” 命令查詢。通過 NMT 工具,我們可以快速區(qū)分內(nèi)存泄露是否源自 JVM 分配。

pmap

對于非 JVM 分配的內(nèi)存,經(jīng)常需要用到 pmap 這個工具了,這是一個 linux 系統(tǒng)自帶工具,能夠從系統(tǒng)層面輸出目標(biāo)進(jìn)程內(nèi)存使用的詳細(xì)情況,用法非常簡單:

pmap [參數(shù)] 《pid》

常用的選項(xiàng)是 “-x” 或 “-X”,都是用來控制輸出信息的詳細(xì)程度。

上圖是 pmap 部分輸出信息,每列含義為

pYYBAGFNe46AXEijAACM5oGR0ow649.png

現(xiàn)象:

某業(yè)務(wù)集群中,多個節(jié)點(diǎn)出現(xiàn)業(yè)務(wù)進(jìn)程內(nèi)存消耗緩慢增長現(xiàn)象,以其中一個節(jié)點(diǎn)為例:

3cd60e48-10ac-11ec-8fb8-12bb97331649.png

如圖所示,這個業(yè)務(wù)進(jìn)程當(dāng)前占用了 4.7G 的虛擬內(nèi)存空間,以及 2.2G 的物理內(nèi)存。已知正常狀態(tài)下該業(yè)務(wù)進(jìn)程的物理內(nèi)存占用量不超過 1G。

分析:

使用命令 “jcmdVM.native_memory detail” 可以看到所有受 JVM 監(jiān)控的內(nèi)存分布情況:

3cfaa4e2-10ac-11ec-8fb8-12bb97331649.png

上圖只是截取了 nmt(Native Memory Tracking) 命令展示的概覽信息,這個業(yè)務(wù)進(jìn)程占用的 2.2G 物理內(nèi)存中,受 JVM 監(jiān)控的大概只占了 0.7G(上圖中的 committed),意味著有 1.5G 物理內(nèi)存不受 JVM 管控。JVM 可以監(jiān)控到 Java 堆、元空間、CodeCache、直接內(nèi)存等區(qū)域,但無法監(jiān)控到那些由 JVM 之外的 Native Code 申請的內(nèi)存,例如典型的場景:第三方 so 庫中調(diào)用 malloc 函數(shù)申請一塊內(nèi)存的行為無法被 JVM 感知到。

nmt 除了會展示概覽之外,還會詳細(xì)羅列每一片受 JVM 監(jiān)控的內(nèi)存,包括其地址,將這些 JVM 監(jiān)控到的內(nèi)存布局和用 pmap 得到的完整的進(jìn)程內(nèi)存布局做一個對比篩查,這里忽略 nmt 和 pmap(下圖 pmap 命令中 25600 是進(jìn)程號)詳細(xì)內(nèi)存地址的信息,直接給出最可疑的那塊內(nèi)存:

3d0e578a-10ac-11ec-8fb8-12bb97331649.png

由圖可知,這片 1.7G 左右的內(nèi)存區(qū)域?qū)儆谙到y(tǒng)層面的堆區(qū)。

備注:這片系統(tǒng)堆區(qū)之所以稍大于上面計(jì)算得到的差值,原因大概是 nmt 中顯示的 committed 內(nèi)存并不對應(yīng)真正占用的物理內(nèi)存(linux 使用 Lazy 策略管理進(jìn)程內(nèi)存),實(shí)際通常會稍小。

系統(tǒng)堆區(qū)主要就是由 libc 庫接口 malloc 申請的內(nèi)存組合而成,所以接下來就是去跟蹤業(yè)務(wù)進(jìn)程中的每次 malloc 調(diào)用,可以借助 GDB:

3d235a5e-10ac-11ec-8fb8-12bb97331649.png

實(shí)際上會有大量的干擾項(xiàng),這些干擾項(xiàng)一方面來自 JVM 內(nèi)部,比如:

3d3cc782-10ac-11ec-8fb8-12bb97331649.png

這部分干擾項(xiàng)很容易被排除,凡是調(diào)用棧中存在 “os::malloc” 這個棧幀的干擾項(xiàng)就可以直接忽視,因?yàn)檫@些 malloc 行為都會被 nmt 監(jiān)控到,而上面已經(jīng)排除了受 JVM 監(jiān)控內(nèi)存泄漏的可能。

另一部分干擾項(xiàng)則來自 JDK,比如:

3d6b9ee0-10ac-11ec-8fb8-12bb97331649.png

有如上圖所示,不少 JDK 的本地方法中直接或間接調(diào)用了 malloc,這部分 malloc 行為通常是不受 JVM 監(jiān)控的,所以需要根據(jù)具體情況逐個排查,還是以上圖為例,排查過程如下:

3dac2b22-10ac-11ec-8fb8-12bb97331649.png

注意圖中臨時中斷的值(0x0000ffff5fc55d00)來自于第一個中斷 b malloc 中斷發(fā)生后的結(jié)果。

這里稍微解釋一下上面 GDB 在做的排查過程,就是檢查 malloc 返回的內(nèi)存地址后續(xù)是否有通過 free 釋放(通過 tb free if X3 這個命令,具體用法可以參考 GDB 調(diào)試),顯然在這個例子中是有釋放的。

通過這種排查方式,幾經(jīng)篩選,最終找到了一個可疑的 malloc 場景:

3dbaf18e-10ac-11ec-8fb8-12bb97331649.png

從調(diào)用棧信息可以知道,這是一個 JDK 中的本地方法 sun.nio.fs.UnixNativeDispatcher.opendir0,作用是打開一個目錄,但后續(xù)始終沒有進(jìn)行關(guān)閉操作。進(jìn)一步分析可知,該可疑 opendir 操作會周期性執(zhí)行,而且都是操作同一個目錄 “/xxx/nginx/etc/nginx/conf”,看來,是有個業(yè)務(wù)線程在定時訪問 nginx 的配置目錄,每次訪問完卻沒有關(guān)閉打開的目錄。

分析到這里,其實(shí)這個問題已經(jīng)差不多水落石出。和業(yè)務(wù)方確認(rèn),存在一個定時器線程在周期性讀取 nginx 的配置文件,代碼大概是這樣子的:

3dfba080-10ac-11ec-8fb8-12bb97331649.png

翻了一下相關(guān) JDK 源碼,F(xiàn)iles.list 方法是有在末尾注冊一個關(guān)閉鉤子的:

3e0bb2b8-10ac-11ec-8fb8-12bb97331649.png

也就是說,F(xiàn)iles.list 方法返回的目錄資源是需要手動釋放的,否則就會發(fā)生資源泄漏。

由于這個目錄資源底層是會關(guān)聯(lián)一個 fd 的,所以泄漏問題還可以通過另一個地方進(jìn)行佐證:

3e3fafaa-10ac-11ec-8fb8-12bb97331649.png

該業(yè)務(wù)進(jìn)程目前已經(jīng)消耗了 51116 個 fd!

假設(shè)這些 fd 都是 opendir 關(guān)聯(lián)的,每個 opendir 消耗 32K,則總共消耗 1.6G,顯然可以跟上面泄漏的內(nèi)存值基本對上。

總結(jié):

稍微了解了一下,發(fā)現(xiàn)幾乎沒人知道 JDK 方法 Files.list 是需要關(guān)閉的,這個案例算是給大家都提了個醒。

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11232

    瀏覽量

    208961
  • 源碼
    +關(guān)注

    關(guān)注

    8

    文章

    633

    瀏覽量

    29147
  • JVM
    JVM
    +關(guān)注

    關(guān)注

    0

    文章

    157

    瀏覽量

    12212
  • JDK
    JDK
    +關(guān)注

    關(guān)注

    0

    文章

    81

    瀏覽量

    16579

原文標(biāo)題:使用 NMT 和 pmap 解決 JVM 資源泄漏問題

文章出處:【微信號:openEulercommunity,微信公眾號:openEuler】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    聊聊JVM如何優(yōu)化

    首先應(yīng)該明確的是JVM調(diào)優(yōu)不是常規(guī)手段,JVM的存在本身就是為了減輕開發(fā)對于內(nèi)存管理的負(fù)擔(dān),當(dāng)出現(xiàn)性能問題的時候第一時間考慮的是代碼邏輯與設(shè)計(jì)方案,以及是否達(dá)到依賴中間件的瓶頸,最后才是針對JVM
    的頭像 發(fā)表于 08-05 17:49 ?443次閱讀
    聊聊<b class='flag-5'>JVM</b>如何優(yōu)化

    eclipse設(shè)置jvm內(nèi)存大小

    內(nèi)存大小,并對其背后的原理進(jìn)行解釋。 JVM(Java虛擬機(jī))是Java程序的運(yùn)行環(huán)境,它負(fù)責(zé)將Java字節(jié)碼翻譯成機(jī)器碼,以便在不同的平臺上執(zhí)行。JVM使用內(nèi)存存儲運(yùn)行時對象和執(zhí)行過程中的臨時數(shù)據(jù)。如果
    的頭像 發(fā)表于 12-06 11:43 ?1831次閱讀

    weblogic設(shè)置jvm內(nèi)存大小

    如何設(shè)置WebLogic服務(wù)器的JVM內(nèi)存大小。 一、了解JVM內(nèi)存 JVM(Java Virtual Machine)是Java應(yīng)用程序的運(yùn)行環(huán)境。JVM使用一個被稱為堆(Heap)
    的頭像 發(fā)表于 12-05 14:44 ?2981次閱讀

    weblogic jvm參數(shù)配置

    在WebLogic中,JVM參數(shù)配置是非常重要的,它可以對應(yīng)用程序的性能和穩(wěn)定性產(chǎn)生直接影響。JVM參數(shù)通過調(diào)整Java虛擬機(jī)的運(yùn)行時行為,可以優(yōu)化內(nèi)存管理、垃圾回收以及線程管理等方面的性能。 首先
    的頭像 發(fā)表于 12-05 14:31 ?1374次閱讀

    jvm和jmm的區(qū)別

    JVM(Java Virtual Machine)和JMM(Java Memory Model)是 Java 開發(fā)者非常熟悉的概念。JVM 是 Java 程序的運(yùn)行環(huán)境,而 JMM 則定義了多線程
    的頭像 發(fā)表于 12-05 14:27 ?1300次閱讀

    jvm配置的mx

    JVM配置中的mx參數(shù)主要用于設(shè)置JVM的最大堆內(nèi)存大小。本文將詳細(xì)介紹mx參數(shù)的作用、配置方法以及如何選擇合適的值。 一、mx參數(shù)的作用 在JVM中,堆內(nèi)存用于存放對象實(shí)例以及相關(guān)數(shù)據(jù)。mx參數(shù)
    的頭像 發(fā)表于 12-05 14:24 ?681次閱讀

    jvm配置metaspace最大值的參數(shù)

    堆內(nèi)存限制):該參數(shù)用于設(shè)置JVM堆的最大大小。在JVM啟動時,可以使用以下命令配置Metaspace的最大大小: java -Xmx ... 其中,``可以是一些表示大小的標(biāo)記
    的頭像 發(fā)表于 12-05 14:21 ?2041次閱讀

    jvm調(diào)優(yōu)工具有哪些

    、基于GUI的監(jiān)控和故障排查工具,提供了對JVM各種資源的可視化監(jiān)控和分析,例如CPU使用率、內(nèi)存使用情況、線程狀態(tài)等。可以通過JMX(Java Management Extensions)連接和監(jiān)控
    的頭像 發(fā)表于 12-05 11:44 ?1035次閱讀

    jvm參數(shù)的設(shè)置和jvm調(diào)優(yōu)

    JVM(Java虛擬機(jī))參數(shù)的設(shè)置和調(diào)優(yōu)對于提高Java應(yīng)用程序的性能和穩(wěn)定性非常重要。在本文中,我們將詳細(xì)介紹JVM參數(shù)的設(shè)置和調(diào)優(yōu)方法。 一、JVM參數(shù)的設(shè)置 內(nèi)存參數(shù): -Xms:設(shè)置J
    的頭像 發(fā)表于 12-05 11:36 ?1444次閱讀

    jvm調(diào)優(yōu)參數(shù)

    和類元數(shù)據(jù)等方面的參數(shù)設(shè)置。下面我們將詳細(xì)介紹這些參數(shù)以及如何進(jìn)行優(yōu)化。 首先,堆內(nèi)存是JVM中用于存放對象實(shí)例的內(nèi)存區(qū)域。通過調(diào)整堆內(nèi)存的大小,我們可以控制應(yīng)用程序?qū)?nèi)存資源的使用。JVM的堆內(nèi)存包括新生代和老年代兩部分。新生
    的頭像 發(fā)表于 12-05 11:29 ?603次閱讀

    什么場景需要jvm調(diào)優(yōu)

    JVM調(diào)優(yōu)是指對Java虛擬機(jī)進(jìn)行性能優(yōu)化和資源管理,以提高應(yīng)用程序的運(yùn)行效率和吞吐量。JVM調(diào)優(yōu)的場景有很多,下面將詳細(xì)介紹各種不同的場景。 高并發(fā)場景:在高并發(fā)場景下,系統(tǒng)需要處理大量的并發(fā)請求
    的頭像 發(fā)表于 12-05 11:14 ?1381次閱讀

    jvm內(nèi)存模型和內(nèi)存結(jié)構(gòu)

    JVM(Java虛擬機(jī))是Java程序的運(yùn)行平臺,它負(fù)責(zé)將Java程序轉(zhuǎn)換成機(jī)器碼并在計(jì)算機(jī)上執(zhí)行。在JVM中,內(nèi)存模型和內(nèi)存結(jié)構(gòu)是兩個重要的概念,本文將詳細(xì)介紹它們。 一、JVM內(nèi)存模型 J
    的頭像 發(fā)表于 12-05 11:08 ?897次閱讀

    jvm內(nèi)存分析命令和工具

    JVM內(nèi)存分析是Java開發(fā)和調(diào)優(yōu)過程中非常重要的一部分。通過對JVM內(nèi)存分析命令和工具的深入了解和使用,可以幫助開發(fā)人員識別內(nèi)存泄漏、性能瓶頸等問題,并對Java應(yīng)用進(jìn)行優(yōu)化。 下面將從不同的角度
    的頭像 發(fā)表于 12-05 11:07 ?1132次閱讀

    jvm內(nèi)存溢出該如何定位解決

    在Java應(yīng)用程序中,JVM(Java虛擬機(jī))內(nèi)存溢出是指Java應(yīng)用程序試圖分配的內(nèi)存超過了JVM所允許的最大內(nèi)存大小,導(dǎo)致程序無法正常執(zhí)行。內(nèi)存溢出通常是由以下幾個原因引起的:內(nèi)存泄漏、對象大小
    的頭像 發(fā)表于 12-05 11:05 ?1295次閱讀

    jvm的dump太大了怎么分析

    文件需要耗費(fèi)大量的時間和計(jì)算資源。 然而,這并不意味著我們無法分析和利用JVM dump文件。以下是一些方法和技巧,可幫助我們有效地分析大型JVM dump文件。 使用工具:首先,我們可以使用一些專門用于分析
    的頭像 發(fā)表于 12-05 11:01 ?2507次閱讀