簡介
Java內(nèi)存回收機(jī)制 不論哪種語言的內(nèi)存分配方式甚带,都需要返回所分配內(nèi)存的真實(shí)地址她肯,也就是返回一個指針到內(nèi)存塊的首地址佳头。Java中對象是采用new或者反射的方法創(chuàng)建的,這些對象的創(chuàng)建都是在堆(Heap)中分配的辕宏,所有對象的回收都是由Java虛擬機(jī)通過垃圾回收機(jī)制完成的畜晰。GC為了能夠正確釋放對象,會監(jiān)控每個對象的運(yùn)行狀況瑞筐,對他們的申請凄鼻、引用、被引用聚假、賦值等狀況進(jìn)行監(jiān)控块蚌,Java會使用有向圖的方法進(jìn)行管理內(nèi)存,實(shí)時監(jiān)控對象是否可以達(dá)到膘格,如果不可到達(dá)峭范,則就將其回收,這樣也可以消除引用循環(huán)的問題瘪贱。在Java語言中纱控,判斷一個內(nèi)存空間是否符合垃圾收集標(biāo)準(zhǔn)有兩個:一個是給對象賦予了空值null,以下再沒有調(diào)用過菜秦,另一個是給對象賦予了新值甜害,這樣重新分配了內(nèi)存空間。
Java內(nèi)存泄露引起原因 首先球昨,什么是內(nèi)存泄露尔店?經(jīng)常聽人談起內(nèi)存泄露,但要問什么是內(nèi)存泄露主慰,沒幾個說得清楚嚣州。內(nèi)存泄露是指無用對象(不再使用的對象)持續(xù)占有內(nèi)存或無用對象的內(nèi)存得不到及時釋放,從而造成的內(nèi)存空間的浪費(fèi)稱為內(nèi)存泄露共螺。內(nèi)存泄露有時不嚴(yán)重且不易察覺该肴,這樣開發(fā)者就不知道存在內(nèi)存泄露,但有時也會很嚴(yán)重藐不,會提示你Out of memory匀哄。那么,Java內(nèi)存泄露根本原因是什么呢佳吞?長生命周期的對象持有短生命周期對象的引用就很可能發(fā)生內(nèi)存泄露拱雏,盡管短生命周期對象已經(jīng)不再需要棉安,但是因?yàn)殚L生命周期對象持有它的引用而導(dǎo)致不能被回收底扳,這就是java中內(nèi)存泄露的發(fā)生場景。
具體主要有如下幾大類:
- 靜態(tài)集合類引起內(nèi)存泄露: 像HashMap贡耽、Vector等的使用最容易出現(xiàn)內(nèi)存泄露衷模,這些靜態(tài)變量的生命周期和應(yīng)用程序一致鹊汛,他們所引用的所有的對象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;
}
在這個例子中刁憋,循環(huán)申請Object 對
象,并將所申請的對象放入一個Vector 中木蹬,如果僅僅釋放引用本身(o=null)至耻,那么Vector 仍然引用該對象,所以這個對象對GC 來說是不可回收的镊叁。因此尘颓,如果對象加入到Vector 后,還必須從Vector 中刪除晦譬,最簡單的方法就是將Vector對象設(shè)置為null疤苹。
- 當(dāng)集合里面的對象屬性被修改后,再調(diào)用remove()方法時不起作用敛腌。
例:
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()+" 個元素!");
//結(jié)果:總共有:3 個元素!
p3.setAge(2);
//修改p3的年齡,此時p3元素對應(yīng)的hashcode值發(fā)生改
變
set.remove(p3);
//此時remove不掉卧土,造成內(nèi)存泄漏
set.add(p3);
//重新添加,居然添加成功
System.out.println("總共有:"+set.size()+" 個元素!");
//結(jié)果:總共有:4 個元素!
for (Person person : set) { System.out.println(person); } }
監(jiān)聽器 在java 編程中像樊,我們都需要和監(jiān)聽器打交道尤莺,通常一個應(yīng)用當(dāng)中會用到很多監(jiān)聽器,我們會調(diào)用一個控件的諸如addXXXListener()等方法來增加監(jiān)聽器凶硅,但往往在釋放對象的時候卻沒有記住去刪除這些監(jiān)聽器缝裁,從而增加了內(nèi)存泄漏的機(jī)會。
各種連接 比如數(shù)據(jù)庫連接(dataSourse.getConnection())足绅,網(wǎng)絡(luò)連接(socket)和io連接捷绑,除非其顯式的調(diào)用了其close()方法將其連接關(guān)閉,否則是不會自動被GC 回收的氢妈。對于Resultset 和Statement 對象可以不進(jìn)行顯式回收粹污,但Connection 一定要顯式回收,因?yàn)镃onnection 在任何時候都無法自動回收首量,而Connection一旦回收壮吩,Resultset 和Statement 對象就會立即為NULL。但是如果使用連接池加缘,情況就不一樣了鸭叙,除了要顯式地關(guān)閉連接,還必須顯式地關(guān)閉Resultset Statement 對象(關(guān)閉其中一個拣宏,另外一個也會關(guān)閉)沈贝,否則就會造成大量的Statement 對象無法釋放,從而引起內(nèi)存泄漏勋乾。這種情況下一般都會在try里面去的連接宋下,在finally里面釋放連接嗡善。
5、內(nèi)部類和外部模塊等的引用 內(nèi)部類的引用是比較容易遺忘的一種学歧,而且一旦沒釋放可能導(dǎo)致一系列的后繼類對象沒有釋放罩引。此外程序員還要小心外部模塊不經(jīng)意的引用,例如程序員A 負(fù)責(zé)A 模塊枝笨,調(diào)用了B 模塊的一個方法如: public void registerMsg(Object b); 這種調(diào)用就要非常小心了袁铐,傳入了一個對象,很可能模塊B就保持了對該對象的引用横浑,這時候就需要注意模塊B 是否提供相應(yīng)的操作去除引用昭躺。單例模式 不正確使用單例模式是引起內(nèi)存泄露的一個常見問題,單例對象在被初始化后將在JVM的整個生命周期中存在(以靜態(tài)變量的方式)伪嫁,如果單例對象持有外部對象的引用领炫,那么這個外部對象將不能被jvm正常回收张咳,導(dǎo)致內(nèi)存泄露帝洪,考慮下面的例子: