掃二維碼與項(xiàng)目經(jīng)理溝通
我們?cè)谖⑿派?4小時(shí)期待你的聲音
解答本文疑問/技術(shù)咨詢/運(yùn)營(yíng)咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流
Java內(nèi)存泄漏一直Java程序中最常見的問題之一,它會(huì)導(dǎo)致內(nèi)存溢出,最終導(dǎo)致程序崩潰。我們可能對(duì)內(nèi)存泄漏很熟悉,但又不是那么熟悉,真的遇到事故的時(shí)候,內(nèi)存泄漏問題排查起來卻也沒有那么容易。本篇就再次梳理一下Java內(nèi)存泄漏的那些事。

使用Java編寫程序時(shí),我們使用new關(guān)鍵字創(chuàng)建對(duì)象。而且我們還不需要專門在對(duì)象使用完成后去釋放其占用的內(nèi)存,這是因?yàn)镴ava有專門的垃圾回收器來負(fù)責(zé)刪除不需要的對(duì)象。只要不被使用的對(duì)象有垃圾回收器回收,那么程序會(huì)處于正常運(yùn)行的狀態(tài),但是垃圾回收器無法刪除那些不被使用的對(duì)象時(shí),我們的Java程序則可能發(fā)生了內(nèi)存泄漏。
內(nèi)存泄漏指的是JVM中某些不再需要使用的對(duì)象,仍然存活于JVM中而不能及時(shí)釋放而導(dǎo)致內(nèi)存空間的浪費(fèi)。Java中內(nèi)存泄漏的原因有多種,這些眾多的因素會(huì)導(dǎo)致Java程序產(chǎn)生不同類型的內(nèi)存泄漏,隨著時(shí)間的推移,內(nèi)存泄漏會(huì)使程序增加額外的內(nèi)存資源占用,從而導(dǎo)致程序性能下降。
垃圾回收器會(huì)回收長(zhǎng)時(shí)間沒有引用的對(duì)象,但是它不會(huì)回收那些還存在引用的對(duì)象,這就是產(chǎn)生內(nèi)存泄漏的原因。
所以為了防止內(nèi)存泄漏,程序設(shè)計(jì)之初就需要考慮去釋放那些不使用的內(nèi)存空間,而開發(fā)人員也應(yīng)當(dāng)時(shí)刻考慮內(nèi)存泄漏的可能性,并增加一些測(cè)試和檢測(cè)避免內(nèi)存泄漏。
Java中,我們可能會(huì)遇到棧內(nèi)存泄露和堆內(nèi)存泄漏。
其中堆內(nèi)存泄漏是由于創(chuàng)建后的對(duì)象一直存在于堆中,不再需要的對(duì)象其引用一直沒有被移除。這些無用的對(duì)象會(huì)慢慢占用內(nèi)存,最后導(dǎo)致內(nèi)存溢出。
棧內(nèi)存泄漏由于方法不斷被調(diào)用,但是一直沒有退出方法。這種情況可能發(fā)生在無限循環(huán)或遞歸掉用時(shí),最終導(dǎo)致棧內(nèi)存溢出。
Java中內(nèi)存泄漏主要是因?yàn)椴荒苷_釋放不需要的資源,長(zhǎng)生命周期對(duì)象持有短生命周期對(duì)象的引用。
靜態(tài)字段引起的內(nèi)存泄漏比較常見,如果某個(gè)不需要的類中含有靜態(tài)字段,那么就會(huì)造成內(nèi)存泄漏。單例模式中如果持有其他的類引用就會(huì)造成內(nèi)存泄漏,靜態(tài)集合如HashMap,LinkedList等持有的一些對(duì)象沒有及時(shí)釋放等。
threadlocal引用一個(gè)對(duì)象使用完成后并沒有被及時(shí)remove掉,線程一直存活的情況下(使用線程池時(shí))就會(huì)發(fā)生內(nèi)存泄漏。
大多時(shí)候內(nèi)存泄漏都是由于開發(fā)人員的代碼錯(cuò)誤導(dǎo)致的,要防止這種內(nèi)存泄漏,就需要編寫必要的代碼來配合垃圾回收器釋放資源。
內(nèi)存泄漏很難定位并修復(fù),但是我們可以遵循以下幾個(gè)步驟去定位并修復(fù):

我們?cè)谖⑿派?4小時(shí)期待你的聲音
解答本文疑問/技術(shù)咨詢/運(yùn)營(yíng)咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流