更新時間:2022-12-29 11:44:47 來源:動力節點 瀏覽1667次
大家在進行Java開發的時候會遇到Java內存泄露的情況,那么,Java內存泄露的原因有哪些?下面來我們就來給大家講解一下。
每當創建連接或者打開流時,JVM都會為這些資源分配內存。如果沒有關閉連接,會導致持續占有內存。在任意情況下,資源留下的開放連接都會消耗內存,如果我們不處理,就會降低性能,甚至OOM。
解決辦法:使用finally塊關閉資源;關閉資源的代碼,不應該有異常;jdk1.7后,可以使用try-with-resource塊。
大量使用static字段會潛在的導致內存泄露,在Java中,靜態字段通常擁有與整個應用程序相匹配的生命周期。
解決辦法:最大限度的減少靜態變量的使用;單例模式時,依賴于延遲加載對象而不是立即加載方式。
非靜態內部類的初始化,總是需要外部類的實例;默認情況下,每個非靜態內部類都包含對其包含內的隱式引用,如果我們在應用程序中使用這個內部類對象,那么即使在我們的包含類對象超出范圍后,它也不會被垃圾收集。
解決辦法:如果內部類不需要訪問包含的類成員,考慮轉換為靜態類。
在HashMap和HashSet這種集合中,常常用到equal()和hashCode()來比較對象,如果重寫不合理,將會成為潛在的內存泄露問題。
解決辦法:用最佳的方式重寫equals()和hashCode。
如果我們讀取一個很大的String對象,并調用了inter(),那么它將放到字符串池中,位于PermGen中,只要應用程序運行,該字符串就會保留,這就會占用內存,可能造成OOM。
解決辦法:增加PermGen的大小,-XX:MaxPermSize=512m;升級Java版本,JDK1.7后字符串池轉移到了堆中。
使用ThreadLocal時,每個線程只要處于存貨狀態就可保留對其ThreadLocal變量副本的隱式調用,且將保留其自己的副本。使用不當,就會引起內存泄露。
一旦線程不在存在,ThreadLocals就應該被垃圾收集,而現在線程的創建都是使用線程池,線程池有線程重用的功能,因此線程就不會被垃圾回收器回收。所以使用到ThreadLocals來保留線程池中線程的變量副本時,ThreadLocals沒有顯示的刪除時,就會一直保留在內存中,不會被垃圾回收。
解決辦法:不在使用ThreadLocal時,調用remove()方法,該方法刪除了此變量的當前線程值。不要使用ThreadLocal.set(null),它只是查找與當前線程關聯的Map并將鍵值對設置為當前線程為null。
重寫finalize()方法時,該類的對象不會立即被垃圾收集器收集,如果finalize()方法的代碼有問題,那么會潛在的引發OOM;
解決辦法:避免重寫finalize()。
java內存泄漏是java常見異常的一種,我們遇到這種情況的時候,需要及時進行排查,知道錯誤的源頭才能進行快速解決!
0基礎 0學費 15天面授
有基礎 直達就業
業余時間 高薪轉行
工作1~3年,加薪神器
工作3~5年,晉升架構
提交申請后,顧問老師會電話與您溝通安排學習