Java詳解線(xiàn)上內(nèi)存暴漲問(wèn)題定位和解決方案
前因:
因?yàn)镽EST規(guī)范,定義資源獲取接口使用GET請(qǐng)求,參數(shù)拼接在url上。
如果按上述定義,當(dāng)參數(shù)過(guò)長(zhǎng),超過(guò)tomcat默認(rèn)配置 max-http-header-size :8kb
會(huì)報(bào)一下錯(cuò)誤信息:
Request header is too large
可以修改springboot配置,調(diào)整請(qǐng)求頭大小
server:
max-http-header-size: xxx
后果:
如果max-http-header-size設(shè)置過(guò)大,會(huì)導(dǎo)致接口吞吐下降,jvm oom,內(nèi)存泄漏。
因?yàn)閠omcat 會(huì)用HeapByteBuffer 預(yù)分配請(qǐng)求頭內(nèi)存大小,在堆上分配。

請(qǐng)求和響應(yīng)都是一樣的配置,每次請(qǐng)求處理預(yù)先分配,2倍配置值內(nèi)存大小在 jvm 堆中
請(qǐng)求過(guò)多,導(dǎo)致線(xiàn)上內(nèi)存暴漲,老年代有3GB多。使用jmap dump線(xiàn)上內(nèi)存數(shù)據(jù),使用 JProfiler 分析。


符合配置大小和源碼對(duì)象

數(shù)組有3GB,和老年代和eden區(qū)總和大小相近。
-XX:PretenureSizeThreshold jvm參數(shù)用來(lái)設(shè)置默認(rèn)值,當(dāng)數(shù)組或?qū)ο蟠笮〕^(guò)這個(gè)設(shè)定值,直接在 Old Gen 老年代分配;默認(rèn)值0,當(dāng)超過(guò)eden區(qū)的大小的時(shí)候,直接分配到old區(qū)。
使用 java -XX:+PrintCommandLineFlags -version

發(fā)現(xiàn)并沒(méi)有使用
-XX:PretenureSizeThreshold參數(shù),所以是
max-http-header-size設(shè)置過(guò)大,eden區(qū)分配不夠,直接分配到old區(qū),堆區(qū)內(nèi)存不夠,自動(dòng)擴(kuò)容,導(dǎo)致old區(qū)數(shù)據(jù)越來(lái)越多,頻繁觸發(fā)FullGC。
JVM初始分配的內(nèi)存由-Xms指定,默認(rèn)是物理內(nèi)存的1/64;JVM最大分配的內(nèi)存由-Xmx指定,默認(rèn)是物理內(nèi)存的1/4。默認(rèn)空余堆內(nèi)存小 于40%時(shí),JVM就會(huì)增大堆直到-Xmx的最大限制;空余堆內(nèi)存大于70%時(shí),JVM會(huì)減少堆直到-Xms的最小限制。因此服務(wù)器一般設(shè)置-Xms、 -Xmx相等以避免在每次GC后調(diào)整堆的大小。



其中http-nio-9005-exec-線(xiàn)程有146個(gè),和前面配置需要分配的20M內(nèi)存請(qǐng)求,相乘的數(shù)據(jù)與3000MB相近。

tomcat任務(wù)線(xiàn)程池 最大線(xiàn)程數(shù)200,存活時(shí)間60s

因?yàn)門(mén)askQueue 重寫(xiě)了offer方法,在線(xiàn)程池大小小于最大線(xiàn)程數(shù)時(shí),任務(wù)不會(huì)放入任務(wù)隊(duì)列,只會(huì)交給現(xiàn)有線(xiàn)程執(zhí)行;存活時(shí)間60s,只有當(dāng)線(xiàn)程空閑60s才會(huì)被回收,也就是**60秒內(nèi)請(qǐng)求要小于當(dāng)前線(xiàn)程數(shù),**才會(huì)有空閑線(xiàn)程。這就導(dǎo)致了線(xiàn)程不能及時(shí)被回收。請(qǐng)求數(shù)下降,但是內(nèi)存還是居高不下。
解決方案:
max-http-header-size修改為默認(rèn)值,接口請(qǐng)求方式修改為POST,請(qǐng)求參數(shù)放置于body
到此這篇關(guān)于Java詳解線(xiàn)上內(nèi)存暴漲問(wèn)題定位和解決方案的文章就介紹到這了,更多相關(guān)Java 內(nèi)存暴漲內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot中關(guān)于static和templates的注意事項(xiàng)以及webjars的配置
今天小編就為大家分享一篇關(guān)于SpringBoot中關(guān)于static和templates的注意事項(xiàng)以及webjars的配置,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-01-01
Java?循環(huán)隊(duì)列/環(huán)形隊(duì)列的實(shí)現(xiàn)流程
循環(huán)隊(duì)列又叫環(huán)形隊(duì)列,是一種特殊的隊(duì)列。循環(huán)隊(duì)列解決了隊(duì)列出隊(duì)時(shí)需要將所有數(shù)據(jù)前移一位的問(wèn)題。本文將帶大家詳細(xì)了解循環(huán)隊(duì)列如何實(shí)現(xiàn),需要的朋友可以參考一下2022-02-02
JAVA封裝多線(xiàn)程實(shí)現(xiàn)的方式及原理
這篇文章主要介紹了Java中封裝多線(xiàn)程的原理和常見(jiàn)方式,通過(guò)封裝可以簡(jiǎn)化多線(xiàn)程的使用,提高安全性,并增強(qiáng)代碼的可維護(hù)性和可擴(kuò)展性,需要的朋友可以參考下2025-03-03
Java線(xiàn)程實(shí)現(xiàn)時(shí)間動(dòng)態(tài)顯示
這篇文章主要為大家詳細(xì)介紹了Java線(xiàn)程實(shí)現(xiàn)時(shí)間動(dòng)態(tài)顯示,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-04-04
SpringBoot集成本地緩存性能之王Caffeine示例詳解
這篇文章主要為大家介紹了SpringBoot集成本地緩存性能之王Caffeine的示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07

