java對象分配與回收

首先談談hotspot JVM中內存模型,通常大致分為1.PC程序計數(shù)器區(qū): 用于保存所有線程的PC2.棧區(qū): 每個線程的棧都會分配在這塊空間, 方法中的參數(shù)、局部變量和返回地址調用方法時在這塊區(qū)域分配、方法執(zhí)行完畢時回收3.方法區(qū): 用于保存類的類型、方法、靜態(tài)變量和字符串等常量4.堆區(qū): 用于分配對象和保存對象的引用樹等相關信息通常JVM對象回收主要在堆區(qū),方法區(qū)也會有少量回收
java對象的生命周期和持有該對象引用的變量作用范圍有一定關系。下面分情況討論,下里的引用是指強引用。1.對象被類靜態(tài)變量引用,該靜態(tài)變量保存在方法區(qū),會一直存在,除非把該變量的值改為null, 否則所引用的對象一直存在2.對象被類對象成員變量引用, 成員變量設為null, 就可以被回收3.對象被類對象成員變量ArrayList對象引用, 成員變量設為null, 該對象就可回收,當該對象回收時,ArrayList和ArrayList所包含的對象都會被回收4.對象被方法中的局部變量引用, 當方法執(zhí)行完時,局部變量會被?;厥?,該變量引用的對象就可回收
判斷對象是否可回收,并不是通過大家常說的對象引用計數(shù)法實現(xiàn)的,一般通過根對象(GCRoot)搜索算法,當?shù)礁鶎ο蟛豢蛇_時對象就可回收,所以方法中兩個互相引用的對象是可以被回收的??勺鳛楦鶎ο蟮挠校?.虛擬機棧中引用的對象2.方法區(qū)靜態(tài)類屬性引用的對象3.方法區(qū)常量引用的對象4.本地方法棧中引用的對象
它們的回收時機由一段代碼測試下:
import
java.io.IOException;

import
java.util.ArrayList;

import
java.util.List;

/**

  • java對象回收時機
  • @author cc

*/

class
TestObject {

String a;

TestObject testObj;

public
TestObject(String i){

a = i;

}

/**

  • 當一個對象不可達時
  • 如果沒有finalize()方法,第一次System.gc()時,該對象就會被回收
  • 如果有finalize()方法,第一次System.gc()時,會把該對象加入一個Finalizer隊列,之后讓finalizer線程執(zhí)行finalize()
  • 第二次System.gc()時,該對象才會被真正回收
  • 無論如何,finalize()只會被執(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()時會被回收

*/

public
void
localObj2() {

TestObject lObj =
new
TestObject(
"ref by local variable 2"
);

}

public
void
listObj() {

list.add(
new
TestObject(
"ref by member ArrayList"
));

}

/**

  • 兩個對象互相引用,該方法執(zhí)行完后 a和b, 在System.gc()后都會被回收

*/

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()"
);
// 判斷對象是否可回收不是根據(jù)引用計數(shù)法,而是根據(jù)GCRoot是否不可達

System.gc();
// 互相引用的兩個對象也被回收了

Thread.sleep(
500
);
// 調用System.gc()時,JVM并不一定會馬上執(zhí)行回收,需要等一段時間

demo =
null
;
// ObjectLifeCircle 對象被回收時,它持有的ArrayList對象和該ArrayList中包含的TestObject對象都會被回收

System.out.println(
"second gc()"
);

System.gc();

Thread.sleep(
500
);

ObjectLifeCircle.sObj =
null
;
// 被靜態(tài)變量引用的對象,并不會隨著該類的對象被回收而回收

System.out.println(
"third gc()"
);

System.gc();

}

}

執(zhí)行結果: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

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

  • 1. Java基礎部分 基礎部分的順序:基本語法,類相關的語法,內部類的語法,繼承相關的語法,異常的語法,線程的語...
    子非魚_t_閱讀 34,853評論 18 399
  • 1.什么是垃圾回收? 垃圾回收(Garbage Collection)是Java虛擬機(JVM)垃圾回收器提供...
    簡欲明心閱讀 90,393評論 17 311
  • (一)Java部分 1、列舉出JAVA中6個比較常用的包【天威誠信面試題】 【參考答案】 java.lang;ja...
    獨云閱讀 7,279評論 0 62
  • 狂徒之于詩,多而不專,泛而無精。雖揮筆可就,然無甚佳文。文基之貧,固爾所限甚深。宜力改之! 丁酉 九月十一 度士
    竹林醉翁閱讀 176評論 0 0
  • tar -c: 建立壓縮檔案 -x:解壓 -t:查看內容 -r:向壓縮歸檔文件末尾追加文件 -u:更新原壓縮包中的...
    Mango_yes閱讀 579評論 0 0

友情鏈接更多精彩內容