首先談?wù)刪otspot JVM中內(nèi)存模型,通常大致分為1.PC程序計(jì)數(shù)器區(qū): 用于保存所有線程的PC2.棧區(qū): 每個(gè)線程的棧都會(huì)分配在這塊空間谁尸, 方法中的參數(shù)、局部變量和返回地址調(diào)用方法時(shí)在這塊區(qū)域分配纽甘、方法執(zhí)行完畢時(shí)回收3.方法區(qū): 用于保存類的類型良蛮、方法、靜態(tài)變量和字符串等常量4.堆區(qū): 用于分配對(duì)象和保存對(duì)象的引用樹等相關(guān)信息通常JVM對(duì)象回收主要在堆區(qū)悍赢,方法區(qū)也會(huì)有少量回收
java對(duì)象的生命周期和持有該對(duì)象引用的變量作用范圍有一定關(guān)系决瞳。下面分情況討論,下里的引用是指強(qiáng)引用左权。1.對(duì)象被類靜態(tài)變量引用皮胡,該靜態(tài)變量保存在方法區(qū),會(huì)一直存在赏迟,除非把該變量的值改為null, 否則所引用的對(duì)象一直存在2.對(duì)象被類對(duì)象成員變量引用, 成員變量設(shè)為null, 就可以被回收3.對(duì)象被類對(duì)象成員變量ArrayList對(duì)象引用屡贺, 成員變量設(shè)為null, 該對(duì)象就可回收,當(dāng)該對(duì)象回收時(shí)瀑梗,ArrayList和ArrayList所包含的對(duì)象都會(huì)被回收4.對(duì)象被方法中的局部變量引用, 當(dāng)方法執(zhí)行完時(shí)烹笔,局部變量會(huì)被棧回收抛丽,該變量引用的對(duì)象就可回收
判斷對(duì)象是否可回收谤职,并不是通過大家常說的對(duì)象引用計(jì)數(shù)法實(shí)現(xiàn)的,一般通過根對(duì)象(GCRoot)搜索算法亿鲜,當(dāng)?shù)礁鶎?duì)象不可達(dá)時(shí)對(duì)象就可回收允蜈,所以方法中兩個(gè)互相引用的對(duì)象是可以被回收的冤吨。可作為根對(duì)象的有:1.虛擬機(jī)棧中引用的對(duì)象2.方法區(qū)靜態(tài)類屬性引用的對(duì)象3.方法區(qū)常量引用的對(duì)象4.本地方法棧中引用的對(duì)象
它們的回收時(shí)機(jī)由一段代碼測(cè)試下:
import
java.io.IOException;
import
java.util.ArrayList;
import
java.util.List;
/**
- java對(duì)象回收時(shí)機(jī)
- @author cc
*/
class
TestObject {
String a;
TestObject testObj;
public
TestObject(String i){
a = i;
}
/**
- 當(dāng)一個(gè)對(duì)象不可達(dá)時(shí)
- 如果沒有finalize()方法饶套,第一次System.gc()時(shí)漩蟆,該對(duì)象就會(huì)被回收
- 如果有finalize()方法,第一次System.gc()時(shí)妓蛮,會(huì)把該對(duì)象加入一個(gè)Finalizer隊(duì)列怠李,之后讓finalizer線程執(zhí)行finalize()
- 第二次System.gc()時(shí),該對(duì)象才會(huì)被真正回收
- 無論如何蛤克,finalize()只會(huì)被執(zhí)行一次
*/
@Override
protected
void
finalize()
throws
Throwable {
super
.finalize();
System.out.println(
"dead "
+a);
}
}
class
ObjectLifeCircle {
private
TestObject mObj;
public
static
TestObject sObj =
new
TestObject(
" ref by static variable"
);
public
final
TestObject fObj =
new
TestObject(
"ref by final member"
);
public
List<TestObject> list =
new
ArrayList<TestObject>();
public
void
localObj() {
TestObject lObj =
new
TestObject(
"ref by local variable 1 "
);
sObj = lObj;
}
/**
- 該方法執(zhí)行完后lObj捺癞, System.gc()時(shí)會(huì)被回收
*/
public
void
localObj2() {
TestObject lObj =
new
TestObject(
"ref by local variable 2"
);
}
public
void
listObj() {
list.add(
new
TestObject(
"ref by member ArrayList"
));
}
/**
- 兩個(gè)對(duì)象互相引用,該方法執(zhí)行完后 a和b, 在System.gc()后都會(huì)被回收
*/
public
void
refCountTest(){
TestObject a =
new
TestObject(
"ref counter a"
);
TestObject b =
new
TestObject(
"ref counter b"
);
a.testObj = b;
b.testObj = a;
}
}
public
class
ObjectLifeCircleDemo {
public
static
void
main(String[] args)
throws
IOException, InterruptedException {
ObjectLifeCircle demo =
new
ObjectLifeCircle();
demo.listObj();
demo.localObj();
demo.localObj2();
demo.refCountTest();
System.out.println(
"first gc()"
);
// 判斷對(duì)象是否可回收不是根據(jù)引用計(jì)數(shù)法构挤,而是根據(jù)GCRoot是否不可達(dá)
System.gc();
// 互相引用的兩個(gè)對(duì)象也被回收了
Thread.sleep(
500
);
// 調(diào)用System.gc()時(shí)髓介,JVM并不一定會(huì)馬上執(zhí)行回收,需要等一段時(shí)間
demo =
null
;
// ObjectLifeCircle 對(duì)象被回收時(shí)筋现,它持有的ArrayList對(duì)象和該ArrayList中包含的TestObject對(duì)象都會(huì)被回收
System.out.println(
"second gc()"
);
System.gc();
Thread.sleep(
500
);
ObjectLifeCircle.sObj =
null
;
// 被靜態(tài)變量引用的對(duì)象唐础,并不會(huì)隨著該類的對(duì)象被回收而回收
System.out.println(
"third gc()"
);
System.gc();
}
}
執(zhí)行結(jié)果:first gc()dead ref counter bdead ref counter adead ref by local variable 2dead ref by static variablesecond gc()dead ref by member ArrayListdead ref by final memberthird gc()dead ref by local variable 1