Java內存回收機制
不論哪種語言的內存分配方式泼返,都需要返回所分配內存的真實地址娜氏,也就是返回一個指針到內存塊的首地址抽活。Java中對象是采用new或者反射的方法創(chuàng)建的子巾,這些對象的創(chuàng)建都是在堆(Heap)中分配的,所有對象的回收都是由Java虛擬機通過垃圾回收機制完成的茶敏。GC為了能夠正確釋放對象壤靶,會監(jiān)控每個對象的運行狀況,對他們的申請惊搏、引用贮乳、被引用、賦值等狀況進行監(jiān)控恬惯,Java會使用有向圖的方法進行管理內存向拆,實時監(jiān)控對象是否可以達到,如果不可到達酪耳,則就將其回收浓恳,這樣也可以消除引用循環(huán)的問題。在Java語言中,判斷一個內存空間是否符合垃圾收集標準有兩個:一個是給對象賦予了空值null颈将,以下再沒有調用過梢夯,另一個是給對象賦予了新值,這樣重新分配了內存空間吆鹤。
Java內存泄漏引起的原因
內存泄漏是指無用對象(不再使用的對象)持續(xù)占有內存或無用對象的內存得不到及時釋放厨疙,從而造成內存空間的浪費稱為內存泄漏洲守。內存泄露有時不嚴重且不易察覺疑务,這樣開發(fā)者就不知道存在內存泄露,但有時也會很嚴重梗醇,會提示你Out of memory知允。
Java內存泄漏的根本原因是什么呢?長生命周期的對象持有短生命周期對象的引用就很可能發(fā)生內存泄漏叙谨,盡管短生命周期對象已經不再需要温鸽,但是因為長生命周期持有它的引用而導致不能被回收,這就是Java中內存泄漏的發(fā)生場景手负。具體主要有如下幾大類:
- 1涤垫、靜態(tài)集合類引起內存泄漏:
像HashMap、Vector等的使用最容易出現內存泄露竟终,這些靜態(tài)變量的生命周期和應用程序一致蝠猬,他們所引用的所有的對象Object也不能被釋放,因為他們也將一直被Vector等引用著统捶。例如
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對象設置為null凯力。
-
2、當集合里面的對象屬性被修改后礼华,再調用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() + " 個元素!"); // 結果:總共有:3 個元素!
p3.setAge(2); // 修改p3的年齡,此時p3元素對應的hashcode值發(fā)生改變set.remove(p3); // 此時remove不掉,造成內存泄漏 set.add(p3); // 重新添加圣絮,居然添加成功 System.out.println("總共有:" + set.size() + " 個元素!"); // 結果:總共有:4 個元素! for (Person person : set) { System.out.println(person); }
}
3祈惶、監(jiān)聽器
在java 編程中,我們都需要和監(jiān)聽器打交道,通常一個應用當中會用到很多監(jiān)聽器捧请,我們會調用一個控件的諸如addXXXListener()等方法來增加監(jiān)聽器凡涩,但往往在釋放對象的時候卻沒有記住去刪除這些監(jiān)聽器,從而增加了內存泄漏的機會疹蛉。4活箕、各種連接
比如數據庫連接(dataSourse.getConnection()),網絡連接(socket)和io連接可款,除非其顯式的調用了其close()方法將其連接關閉育韩,否則是不會自動被GC 回收的。對于Resultset 和Statement 對象可以不進行顯式回收闺鲸,但Connection 一定要顯式回收筋讨,因為Connection 在任何時候都無法自動回收,而Connection一旦回收摸恍,Resultset 和Statement 對象就會立即為NULL悉罕。但是如果使用連接池,情況就不一樣了立镶,除了要顯式地關閉連接壁袄,還必須顯式地關閉Resultset Statement 對象(關閉其中一個,另外一個也會關閉)媚媒,否則就會造成大量的Statement 對象無法釋放嗜逻,從而引起內存泄漏。這種情況下一般都會在try里面去的連接欣范,在finally里面釋放連接变泄。5、內部類和外部模塊的引用
內部類的引用是比較容易遺忘的一種恼琼,而且一旦沒釋放可能導致一系列的后繼類對象沒有釋放妨蛹。此外程序員還要小心外部模塊不經意的引用,例如程序員A負責A模塊晴竞,調用了B模塊的一個方法如: public void registerMsg(Object b); 這種調用就要非常小心了蛙卤,傳入了一個對象,很可能模塊B就保持了對該對象的引用噩死,這時候就需要注意模塊B是否提供相應的操作去除引用颤难。6、單例模式
不正確使用單例模式是引起內存泄漏的一個常見問題已维,單例對象在初始化后將在JVM的整個生命周期中存在(以靜態(tài)變量的方式)行嗤,如果單例對象持有外部的引用,那么這個對象將不能被JVM正扯舛回收栅屏,導致內存泄漏飘千。例如
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模式,它持有一個A對象的引用栈雳,而這個A類的對象將不能被回收护奈。