Java基礎(chǔ)詳解之內(nèi)存泄漏
一、什么是內(nèi)存泄漏
內(nèi)存泄漏是指你向系統(tǒng)申請(qǐng)分配內(nèi)存進(jìn)行使用(new/malloc),然后系統(tǒng)在堆內(nèi)存中給這個(gè)對(duì)象申請(qǐng)一塊內(nèi)存空間,但當(dāng)我們使用完了卻沒有歸系統(tǒng)(delete),導(dǎo)致這個(gè)不使用的對(duì)象一直占據(jù)內(nèi)存單元,造成系統(tǒng)將不能再把它分配給需要的程序。
一次內(nèi)存泄漏的危害可以忽略不計(jì),但是內(nèi)存泄漏堆積則后果很嚴(yán)重,無論多少內(nèi)存,遲早會(huì)被占完,造成內(nèi)存泄漏。
二、Java內(nèi)存泄漏引起的原因
1、靜態(tài)集合類引起內(nèi)存泄漏:
像HashMap、Vector等的使用最容易出現(xiàn)內(nèi)存泄露,這些靜態(tài)變量的生命周期和應(yīng)用程序一致,他們所引用的所有的對(duì)象Object也不能被釋放,因?yàn)樗麄円矊⒁恢北籚ector等引用著。
Static Vector v = new Vector(10); for (int i = 1; i<100; i++) { Object o = new Object(); v.add(o); o = null ; }
在這個(gè)例子中,循環(huán)申請(qǐng)Object 對(duì)象,并將所申請(qǐng)的對(duì)象放入一個(gè)Vector 中,如果僅僅釋放引用本身(o=null),那么Vector 仍然引用該對(duì)象,所以這個(gè)對(duì)象對(duì)GC 來說是不可回收的。因此,如果對(duì)象加入到Vector 后,還必須從Vector 中刪除,最簡單的方法就是將Vector對(duì)象設(shè)置為null。
2、當(dāng)集合里面的對(duì)象屬性被修改后,再調(diào)用remove()方法時(shí)不起作用。
public static void main(String[] args) { Set<Person> set = new HashSet<Person>(); Person p1 = new Person( "唐僧" , "pwd1" , 25 ); Person p2 = new Person( "孫悟空" , "pwd2" , 26 ); Person p3 = new Person( "豬八戒" , "pwd3" , 27 ); set.add(p1); set.add(p2); set.add(p3); System.out.println( "總共有:" +set.size()+ " 個(gè)元素!" ); //結(jié)果:總共有:3 個(gè)元素! p3.setAge( 2 ); //修改p3的年齡,此時(shí)p3元素對(duì)應(yīng)的hashcode值發(fā)生改變 set.remove(p3); //此時(shí)remove不掉,造成內(nèi)存泄漏 set.add(p3); //重新添加,居然添加成功 System.out.println( "總共有:" +set.size()+ " 個(gè)元素!" ); //結(jié)果:總共有:4 個(gè)元素! for (Person person : set) { System.out.println(person); } }
3、監(jiān)聽器
在java編程中,我們都需要和監(jiān)聽器打交道,通常一個(gè)應(yīng)用當(dāng)中會(huì)用到很多監(jiān)聽器,我們會(huì)調(diào)用一個(gè)控件的諸如addXXXListener()等方法來增加監(jiān)聽器,但往往在釋放對(duì)象的時(shí)候卻沒有記住去刪除這些監(jiān)聽器,從而增加了內(nèi)存泄漏的機(jī)會(huì)。
4、各種連接
比如數(shù)據(jù)庫連接(dataSourse.getConnection()),網(wǎng)絡(luò)連接(socket)和io連接,除非其顯式的調(diào)用了其close()方法將其連接關(guān)閉,否則是不會(huì)自動(dòng)被GC 回收的。對(duì)于Resultset 和Statement 對(duì)象可以不進(jìn)行顯式回收,但Connection 一定要顯式回收,因?yàn)镃onnection 在任何時(shí)候都無法自動(dòng)回收,而Connection一旦回收,Resultset 和Statement 對(duì)象就會(huì)立即為NULL。但是如果使用連接池,情況就不一樣了,除了要顯式地關(guān)閉連接,還必須顯式地關(guān)閉Resultset Statement 對(duì)象(關(guān)閉其中一個(gè),另外一個(gè)也會(huì)關(guān)閉),否則就會(huì)造成大量的Statement 對(duì)象無法釋放,從而引起內(nèi)存泄漏。這種情況下一般都會(huì)在try里面去的連接,在finally里面釋放連接。
5、內(nèi)部類和外部模塊的引用
內(nèi)部類的引用是比較容易遺忘的一種,而且一旦沒釋放可能導(dǎo)致一系列的后繼類對(duì)象沒有釋放。此外程序員還要小心外部模塊不經(jīng)意的引用,例如程序員A 負(fù)責(zé)A 模塊,調(diào)用了B 模塊的一個(gè)方法如:
public void registerMsg(Object b);
這種調(diào)用就要非常小心了,傳入了一個(gè)對(duì)象,很可能模塊B就保持了對(duì)該對(duì)象的引用,這時(shí)候就需要注意模塊B 是否提供相應(yīng)的操作去除引用。
6、單例模式
不正確使用單例模式是引起內(nèi)存泄漏的一個(gè)常見問題,單例對(duì)象在初始化后將在JVM的整個(gè)生命周期中存在(以靜態(tài)變量的方式),如果單例對(duì)象持有外部的引用,那么這個(gè)對(duì)象將不能被JVM正常回收,導(dǎo)致內(nèi)存泄漏,考慮下面的例子:
class A{ public A(){ B.getInstance().setA( this ); } .... } //B類采用單例模式 class B{ private A a; private static B instance= new B(); public B(){} public static B getInstance(){ return instance; } public void setA(A a){ this .a=a; } //getter... }
顯然B采用singleton模式,它持有一個(gè)A對(duì)象的引用,而這個(gè)A類的對(duì)象將不能被回收。想象下如果A是個(gè)比較復(fù)雜的對(duì)象或者集合類型會(huì)發(fā)生什么情況。
7、redis緩存雪崩、緩存穿透、緩存擊穿
三、內(nèi)存泄漏的危害
1、頻繁GC:
系統(tǒng)分配給每個(gè)應(yīng)用的內(nèi)存資源都是有限的,內(nèi)存泄漏導(dǎo)致其他組件可用的內(nèi)存變少后,一方面會(huì)使得GC的頻率加劇,再發(fā)生GC的時(shí)候,所有進(jìn)程都必須等待,GC的頻率越高,用戶越容易感應(yīng)到卡頓。另一方面內(nèi)存變少,可能使得系統(tǒng)額外分配給該對(duì)象一些內(nèi)存,而影響整個(gè)系統(tǒng)的運(yùn)行情況。
2、導(dǎo)致程序運(yùn)行崩潰:
一旦內(nèi)存不足以為某些對(duì)象分配所需要的空間,將會(huì)導(dǎo)致程序崩潰,造成體驗(yàn)差。
到此這篇關(guān)于Java基礎(chǔ)詳解之內(nèi)存泄漏的文章就介紹到這了,更多相關(guān)java內(nèi)存泄漏內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot logback如何從apollo配置中心讀取變量
這篇文章主要介紹了springboot logback如何從apollo配置中心讀取變量的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08Java實(shí)現(xiàn)EasyCaptcha圖形驗(yàn)證碼的具體使用
Java圖形驗(yàn)證碼,支持gif、中文、算術(shù)等類型,可用于Java Web、JavaSE等項(xiàng)目,下面就跟隨小編一起來了解一下2021-08-08使用Spring Boot Maven插件的詳細(xì)方法
這篇文章主要介紹了如何使用Spring Boot Maven插件,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-05-05Spring Boot實(shí)戰(zhàn)教程之自動(dòng)配置詳解
Spring Boot的自動(dòng)配置給開發(fā)者帶來了很大的便利,當(dāng)開發(fā)人員在pom文件中添加starter依賴后,maven或者gradle會(huì)自動(dòng)下載很多jar包到classpath中。下面這篇文章主要給大家介紹了關(guān)于Spring Boot自動(dòng)配置的相關(guān)資料,需要的朋友可以參考借鑒,下面來一起看看吧。2017-07-07springboot druid數(shù)據(jù)庫連接池連接失敗后一直重連的解決方法
本文主要介紹了springboot druid數(shù)據(jù)庫連接池連接失敗后一直重連的解決方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04java設(shè)計(jì)模式之代理模式(Porxy)詳解
這篇文章主要為大家詳細(xì)介紹了java設(shè)計(jì)模式之代理模式Porxy的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06詳解SpringBoot靜態(tài)方法獲取bean的三種方式
本文主要介紹了詳解SpringBoot靜態(tài)方法獲取bean的三種方式,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-10-10Spring boot整合shiro+jwt實(shí)現(xiàn)前后端分離
這篇文章主要為大家詳細(xì)介紹了Spring boot整合shiro+jwt實(shí)現(xiàn)前后端分離,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-12-12解決mybatis返回boolean值時(shí)數(shù)據(jù)庫返回null的問題
這篇文章主要介紹了解決mybatis返回boolean值時(shí)數(shù)據(jù)庫返回null的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-11-11HashMap原理及手寫實(shí)現(xiàn)部分區(qū)塊鏈特征
這篇文章主要為大家介紹了HashMap原理及手寫實(shí)現(xiàn)部分區(qū)塊鏈特征,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09