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