Oh My God!e.printStackTrace() 导致系统卡崩
作者:sxgkwei?來源:http://dwz.cn/tQe4fLeD
e.printStackTrace() 會導(dǎo)致鎖死?
這僅僅是打印啊,怎么可能?!
先別驚呼不可能,且聽我細(xì)細(xì)道來。
先看截圖1:
注意右下角區(qū)域,紅框部分。這塊內(nèi)存是什么呢?
非堆!那么,左邊是代碼緩存區(qū)內(nèi)存,右邊紅框就是字符串池,常量,基本類型數(shù)據(jù)的內(nèi)存區(qū)。
然后呢?已經(jīng)滿了。什么原因呢?e.printStackTrace()!
滿了的后果呢?整個(gè)web服務(wù),訪問之后,沒響應(yīng)了,就當(dāng)是卡死掉了。
再來看截圖2:
看看有多少web的請求線程,被卡住在打印這一步!
原因呢?
要打印字符串輸出到控制臺上,那你字符串常量池所在的內(nèi)存塊要有空間啊。
然而,因?yàn)?e.printStackTrace()語句要產(chǎn)生的字符串記錄的是堆棧信息,太長太多,內(nèi)存被填滿了!注意 上面代碼語句:4208行!
來看圖3:
沒毛病,沒沒事兒找事兒冤枉誰。就是這句代碼惹的禍!當(dāng)然,我承認(rèn),被 try 住的代碼本身就有問題,導(dǎo)致很多調(diào)用都會拋異常。
那么,讓我們再來理理整個(gè)事件產(chǎn)生的經(jīng)過:
短時(shí)間內(nèi)大量請求訪問此接口?
-> 代碼本身有問題,很多情況下拋異常?
-> e.printStackTrace() 來打印異常到控制臺
-> 產(chǎn)生錯(cuò)誤堆棧字符串到字符串池內(nèi)存空間
-> 此內(nèi)存空間一下子被占滿了?
-> 開始在此內(nèi)存空間產(chǎn)出字符串的線程還沒完全生產(chǎn)完整,就沒空間了?
-> 大量線程產(chǎn)出字符串產(chǎn)出到一半,等在這兒(等有內(nèi)存了繼續(xù)搞啊)
-> 相互等待,等內(nèi)存,鎖死了,整個(gè)應(yīng)用掛掉了。
綜上,這就是 e.printStackTrace() 引發(fā)的血案。
總結(jié)當(dāng)然重要,有3點(diǎn):
1,代碼質(zhì)量啊親,代碼不拋異常咱不就能愉快的繼續(xù)浪么?
2,不要使用 e.printStackTrace() 啊,這玩意兒,在項(xiàng)目發(fā)布后,除過不斷的刷控制臺,并沒用什么卵用啊,您到是用 log 對象輸出到日志文件里面啊。
3,推及開來,在java中,會產(chǎn)生大量字符串的方法,使用時(shí),一定得悠著點(diǎn),別一不小心撐到肚子(字符串池所屬的那么點(diǎn)非堆內(nèi)存空間),撐到肚子了,會死的啊 。
《新程序員》:云原生和全面數(shù)字化實(shí)踐50位技術(shù)專家共同創(chuàng)作,文字、視頻、音頻交互閱讀總結(jié)
以上是生活随笔為你收集整理的Oh My God!e.printStackTrace() 导致系统卡崩的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Google下的这盘“小”棋
- 下一篇: SpringBoot是如何解析HTTP参