真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

如何理解Java進(jìn)程的OOMKiller

今天就跟大家聊聊有關(guān)如何理解Java進(jìn)程的OOMKiller,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

為巴東等地區(qū)用戶提供了全套網(wǎng)頁設(shè)計制作服務(wù),及巴東網(wǎng)站建設(shè)行業(yè)解決方案。主營業(yè)務(wù)為網(wǎng)站設(shè)計、成都做網(wǎng)站、巴東網(wǎng)站設(shè)計,以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務(wù)。我們深信只要達(dá)到每一位用戶的要求,就會得到認(rèn)可,從而選擇與我們長期合作。這樣,我們也可以走得更遠(yuǎn)!

01 OOM killer

登上機(jī)器后,查看應(yīng)用和中間件日志,確實沒有看到問題。我懷疑是JVM OOM了但沒有配置輸出,正想加上OOM時的堆棧輸出參數(shù),但發(fā)現(xiàn)應(yīng)用啟動命令已經(jīng)包含類似的參數(shù):-XX:+HeapDumpOnOutOfMemoryError -XX:ErrorFile=./hs_err_pid.log -XX:HeapDumpPath=./java_pid.hprof。 看來從應(yīng)用層面查不到什么了,那就再看看系統(tǒng)日志吧。 使用dmesg -T | grep java查看系統(tǒng)日志 如何理解Java進(jìn)程的OOMKiller 果然,從dmesg倒數(shù)第二行返回的信息來看,確實是由于oom導(dǎo)致的,但是這里的oom并非JVM的oom,而是Linux系統(tǒng)的oom。 >Killed process xxx(java), total-vm:7539096kB, anon-rss:3919048kB file-rss:17312kB, shmem-rss:0kB。

在這臺物理內(nèi)存為4GB的機(jī)器上,
`total-vm`是已經(jīng)分配給java進(jìn)程的虛擬內(nèi)存數(shù)量(7539096kb=7.02GB)
`anon-rss`是java進(jìn)程物理內(nèi)存使用量(3919048kB=3.65GB)
`file-rss`是java進(jìn)程映射到設(shè)備或文件系統(tǒng)的使用量(17312kB=16.51MB)

網(wǎng)上已經(jīng)有很多介紹Linux oom killer的文章了,這里只簡單概括:Linux系統(tǒng)在分配物理內(nèi)存時,如果內(nèi)存不足(什么時候不足?)oom killer會選擇oom score得分最高的進(jìn)程殺掉以釋放內(nèi)存。

02 JVM OOM vs. Linux OOM

看起來問題很明確了,是Linux的oom killer殺掉了Java進(jìn)程。但是我還是有個疑問,同樣是OOM,為什么沒有觸發(fā)JVM的OOM呢?帶著這個疑問,繼續(xù)了解了Linux的內(nèi)存分配機(jī)制。

02.1 overcommit與oom killer

以下簡單說一下我的理解,真實分配機(jī)制要復(fù)雜的多。有不對的地方請大家指正。 Linux有一套虛擬內(nèi)存機(jī)制,當(dāng)進(jìn)程向系統(tǒng)申請內(nèi)存時,總體上系統(tǒng)可以有兩種方式答復(fù)進(jìn)程: a.檢查是否真的還有足夠的內(nèi)存(實際值計算一般等于Swap+RAM*overcmmit_ratio)滿足需求,如果有就滿足分配,如果沒有就以分配失敗答復(fù)集成; b.不檢查,直接分配。

這里的分配指的都是進(jìn)程的虛擬內(nèi)存地址的增長。這里的b叫做"Overcommit",也就是發(fā)生了超過系統(tǒng)可接受內(nèi)存范圍的分配。這里的b其實是基于一種樂觀的估計,因為申請的內(nèi)存不一定用到。但是一旦進(jìn)程需要使用所能提供的實際內(nèi)存時,就會導(dǎo)致OOM,此時oom killer就會通過排序oom score犧牲掉一個或多個得分最高的進(jìn)程,以此來釋放內(nèi)存。

再強(qiáng)調(diào)一次理解Overcommit的關(guān)鍵點,內(nèi)存申請不等于內(nèi)存分配,內(nèi)存使用時才真正分配

02.2 JVM OOM

那么JVM的OOM呢?

>One common indication of a memory leak is the java.lang.OutOfMemoryError exception. Usually, this error is thrown when there is insufficient space to allocate an object in the Java heap. In this case, The garbage collector cannot make space available to accommodate a new object, and the heap cannot be expanded further. Also, this error may be thrown when there is insufficient native memory to support the loading of a Java class. In a rare instance, a java.lang.OutOfMemoryError may be thrown when an excessive amount of time is being spent doing garbage collection and little memory is being freed. 摘自O(shè)racle內(nèi)存泄露說明相關(guān)文章

簡單來說,發(fā)生JVM OOM的原因有: Java堆內(nèi)存不足,垃圾回收器不能給新對象騰地兒,堆也無法擴(kuò)展(Java heap space); 本地內(nèi)存不足以支持加載Java類(Metaspace or Perm); 不常見地,過長時間的垃圾回收沒有釋放多少內(nèi)存也會導(dǎo)致OOM(GC Overhead limit exceeded)。 回頭看看,其實是Linux系統(tǒng)把Java進(jìn)程給“騙”了。當(dāng)開啟了允許overcommit的策略時,如果Java進(jìn)程或其他任何進(jìn)程申請了可能過多(超過系統(tǒng)能提供的)虛擬內(nèi)存時,只要系統(tǒng)內(nèi)存還足夠使用,Java進(jìn)程并不會發(fā)生OOM。而當(dāng)Linux系統(tǒng)發(fā)現(xiàn)分配內(nèi)存真的不夠時,就會把oom score最靠前的Java進(jìn)程悄無聲息地干掉(kill -9)

>Java老農(nóng):“爺,給俺批點地,俺要種瓜。” Linux老爺:“爺我高興,給你4096畝地,隨便耍!” Java老農(nóng)種地中...1...2...4....1024... Java老農(nóng):“沒有蟲子騷擾太開心啦,我要這樣到永...” Java卒 Linux老爺:“有沒有人要地呀,老爺我有的是地?。 ?/p>

03 解決方案

分析完原因,其實有很多解決方案。

03.1 申請更大內(nèi)存的機(jī)器

如果有資源,就先嘗試最簡單的辦法吧。

03.2 優(yōu)化代碼

現(xiàn)在的機(jī)器一般都是容器或虛擬機(jī)上只要只跑一個應(yīng)用,如果發(fā)生OOM,最根本的解決之道還是要回歸到代碼上。

03.3 禁止Overcommit

讓系統(tǒng)在申請時就報錯,保守也保險。及早暴露問題,及早進(jìn)行修復(fù)。同時可以修改overcommit_ratio的值改變CommitLimit。 配置參見附錄。

03.4 禁止OOM Killer殺掉關(guān)鍵的應(yīng)用進(jìn)程

對于一些多應(yīng)用進(jìn)程混用的機(jī)器,可以保護(hù)關(guān)鍵進(jìn)程不被kill掉。 sudo echo -1000 > /proc/$pid/oom_score_adj

03.5 禁止OOM Killer

修改panic_on_oom參數(shù)可以關(guān)閉OOM Killer。玩玩還可以,線上系統(tǒng)不建議用,否則系統(tǒng)死給你看。


附錄:

1 Linux的Overcommit的策略

可在/proc/sys/vm/overcommit_memory配置或查看。

>0:默認(rèn)值。啟發(fā)式策略,比較嚴(yán)重的Overcommit將不能滿足,而輕微的Overcommit將被允許。另外,root能Overcommit的值比普通用戶要稍微多些,一般為3%。 1:允許Overcommit,這種策略適合那些不能承受內(nèi)存分配失敗的應(yīng)用,比如某些科學(xué)計算應(yīng)用。(本文中涉及的系統(tǒng)就是開啟的這個策略) 2:禁止Overcommit,在這個情況下,系統(tǒng)所能分配的內(nèi)存不會超過下面的CommitLimit,計算方法為Swap + RAM * /proc/sys/vm/overcmmit_ratio,默認(rèn)50%,可調(diào)整),如果這么多資源已經(jīng)用光,那么后面任何嘗試申請內(nèi)存的行為都會返回錯誤,這通常意味著此時沒法運行任何新程序。

2 查看已分配內(nèi)存

grep -i commit /proc/meminfo
CommitLimit:     6201492 kB    # 虛擬內(nèi)存限定值
Committed_AS:    5770836 kB    # 已分配內(nèi)存,如果大于CommitLimit說明開啟了允許Overcommit的策略

3 查看oom相關(guān)

# 進(jìn)程oom得分,0不kill
cat /proc/{pid}/oom_score 
# 進(jìn)程oom調(diào)整兼容,計算時一般會以oom_score_adj替換
cat /proc/{pid}/oom_adj 
# 用戶打分調(diào)整。最小值-1000,將會禁止oom killer殺此進(jìn)程。
#取值從-1000到1000,表示對最終得分的折扣到懲罰。
cat /proc/{pid}/oom_score_adj

看完上述內(nèi)容,你們對如何理解Java進(jìn)程的OOMKiller有進(jìn)一步的了解嗎?如果還想了解更多知識或者相關(guān)內(nèi)容,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝大家的支持。


網(wǎng)站欄目:如何理解Java進(jìn)程的OOMKiller
轉(zhuǎn)載來源:http://weahome.cn/article/gsicie.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部