作者:sxgkwei
https://my.oschina.net/sxgkwei/blog/825700
e.printStackTrace() 會導致鎖死?
這僅僅是列印啊,怎么可能?!
先別驚呼不可能,且聽我細細道來,
先看截圖1:

注意右下角區域,紅框部分,這塊記憶體是什么呢?
非堆!那么,左邊是代碼快取區記憶體,右邊紅框就是字串池,常量,基本型別資料的記憶體區,
然后呢?已經滿了,什么原因呢?e.printStackTrace()!
滿了的后果呢?整個web服務,訪問之后,沒回應了,就當是卡死掉了,
再來看截圖2:

看看有多少web的請求執行緒,被卡住在列印這一步!
原因呢?
要列印字串輸出到控制臺上,那你字串常量池所在的記憶體塊要有空間啊,
然而,因為 e.printStackTrace()陳述句要產生的字串記錄的是堆疊資訊,太長太多,記憶體被填滿了!注意 上面代碼陳述句:4208行!
來看圖3:

沒毛病,沒沒事兒找事兒冤枉誰,就是這句代碼惹的禍!當然,我承認,被 try 住的代碼本身就有問題,導致很多呼叫都會拋例外,
那么,讓我們再來理理整個事件產生的經過:
短時間內大量請求訪問此介面
-> 代碼本身有問題,很多情況下拋例外
-> e.printStackTrace() 來列印例外到控制臺
-> 產生錯誤堆疊字串到字串池記憶體空間
-> 此記憶體空間一下子被占滿了
-> 開始在此記憶體空間產出字串的執行緒還沒完全生產完整,就沒空間了
-> 大量執行緒產出字串產出到一半,等在這兒(等有記憶體了繼續搞啊)
-> 相互等待,等記憶體,鎖死了,整個應用掛掉了,
綜上,這就是 e.printStackTrace() 引發的血案,
總結當然重要,有3點:
1,代碼質量啊親,代碼不拋例外咱不就能愉快的繼續浪么?
2,不要使用 e.printStackTrace() 啊,這玩意兒,在專案發布后,除過不斷的刷控制臺,并沒用什么卵用啊,您到是用 log 物件輸出到日志檔案里面啊,
3,推及開來,在java中,會產生大量字串的方法,使用時,一定得悠著點,別一不小心撐到肚子(字串池所屬的那么點非堆記憶體空間),撐到肚子了,會死的啊 ,
近期熱文推薦:
1.Java 15 正式發布, 14 個新特性,重繪你的認知!!
2.終于靠開源專案弄到 IntelliJ IDEA 激活碼了,真香!
3.我用 Java 8 寫了一段邏輯,同事直呼看不懂,你試試看,,
4.吊打 Tomcat ,Undertow 性能很炸!!
5.《Java開發手冊(嵩山版)》最新發布,速速下載!
覺得不錯,別忘了隨手點贊+轉發哦!
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/248389.html
標籤:Java
下一篇:Java基礎
