java 中,什么是對象的可達(dá)與不可達(dá)
Java中內(nèi)存泄露,就是因?yàn)閷ο鬅o用卻可達(dá)的原因.
這個細(xì)分下來有三個
1. 不可用不可達(dá)------>這種情況GC會幫我們回收掉,而C++不會
2. 可用可達(dá) ------>正常使用
3. 不可用可達(dá) ------>這種情況會存在內(nèi)存泄露
釋義:
1.不可用不可達(dá)就是我們的變量作用域結(jié)束了,不可用不可達(dá)
**3.不可用可達(dá),就是我們自己沒有將其對象,
舉個例子:
在這個例子中,我們循環(huán)申請Object對象,并將所申請的對象放入一個Vector中,如果我們僅僅釋放引用本身群凶,那么Vector仍然引用該對象,所以這個對象對GC來說是不可回收的。因此螺捐,如果對象加入到Vector后,還必須從Vector中刪除矮燎,最簡單的方法就是將Vector對象設(shè)置為null
Vector v=new Vector(10);
for (int i=1;i<100; i++)
{
Object o=new Object();
v.add(o);
o=null;
}
此時(shí)定血,所有的Object對象都沒有被釋放,因?yàn)樽兞縱引用這些對象诞外。
這時(shí)候這些Object就是不可用可達(dá)的對象,GC不會幫我們清理的, 這就存在了內(nèi)存泄露了.
JVM 怎樣確定一個對象是否可以被回收
比較常被提到的兩種垃圾對象判定算法:
- 列引用計(jì)數(shù)(Reference Counting)
- 可達(dá)性分析(Reachability Analysis)
1.引用計(jì)數(shù)(Reference Counting)
概述:給對象添加一個引用計(jì)數(shù)器澜沟,每有一個地方引用這個對象,計(jì)數(shù)器值加1峡谊,每有一個引用失效則減1茫虽。
應(yīng)用實(shí)例:Python中使用了這種算法判定死對象。
優(yōu)點(diǎn):實(shí)現(xiàn)簡單靖苇、判定效率高
缺點(diǎn):難以解決對象之間的循環(huán)引用問題
2.可達(dá)性分析(Reachability Analysis)
概述:從GC Roots(每種具體實(shí)現(xiàn)對GC Roots有不同的定義)作為起點(diǎn)席噩,向下搜索它們引用的對象,可以生成一棵引用樹贤壁,樹的節(jié)點(diǎn)視為可達(dá)對象悼枢,反之視為不可達(dá)。
應(yīng)用實(shí)例:Java,C#,Lisp都使用這種算法
JVM使用“可達(dá)性分析算法”來判定一個對象是否會可以被回收脾拆,有兩個細(xì)節(jié)需要注意:
Java中GC Roots包括以下幾種對象:
a.虛擬機(jī)棧(幀棧中的本地變量表)中引用的對象
b.方法區(qū)中靜態(tài)屬性引用的對象
c.方法區(qū)中常量引用的對象
d.本地方法棧中JNI引用的對象
2.不可達(dá)對象一定會被回收嗎不是馒索。
執(zhí)行垃圾回收前JVM會執(zhí)行不可達(dá)對象的finalize方法莹妒,如果執(zhí)行完畢之后該對象變?yōu)榭蛇_(dá),則不會被回收它绰上。
但一個對象的finalize方法只會被執(zhí)行一次旨怠。參考資料:《深入理解Java虛擬機(jī)》周志明