java一些小知识点

java 中,什么是对象的可达与不可达

Java中内存泄露,就是因为对象无用却可达的原因.
这个细分下来有三个
1. 不可用不可达------>这种情况GC会帮我们回收掉,而C++不会
2. 可用可达 ------>正常使用
3. 不可用可达 ------>这种情况会存在内存泄露

释义:

1.不可用不可达就是我们的变量作用域结束了,不可用不可达
**3.不可用可达,就是我们自己没有将其对象,
举个例子:
在这个例子中,我们循环申请Object对象,并将所申请的对象放入一个Vector中,如果我们仅仅释放引用本身,那么Vector仍然引用该对象,所以这个对象对GC来说是不可回收的。因此,如果对象加入到Vector后,还必须从Vector中删除,最简单的方法就是将Vector对象设置为null

Vector v=new Vector(10);
for (int i=1;i<100; i++)
{
    Object o=new Object();
    v.add(o);
    o=null; 
} 

此时,所有的Object对象都没有被释放,因为变量v引用这些对象。
这时候这些Object就是不可用可达的对象,GC不会帮我们清理的, 这就存在了内存泄露了.

JVM 怎样确定一个对象是否可以被回收

比较常被提到的两种垃圾对象判定算法:

  1. 列引用计数(Reference Counting)
  2. 可达性分析(Reachability Analysis)

1.引用计数(Reference Counting)
概述:给对象添加一个引用计数器,每有一个地方引用这个对象,计数器值加1,每有一个引用失效则减1。
应用实例:Python中使用了这种算法判定死对象。
优点:实现简单、判定效率高
缺点:难以解决对象之间的循环引用问题

2.可达性分析(Reachability Analysis)

概述:从GC Roots(每种具体实现对GC Roots有不同的定义)作为起点,向下搜索它们引用的对象,可以生成一棵引用树,树的节点视为可达对象,反之视为不可达。
应用实例:Java,C#,Lisp都使用这种算法

JVM使用“可达性分析算法”来判定一个对象是否会可以被回收,有两个细节需要注意:
Java中GC Roots包括以下几种对象:
a.虚拟机栈(帧栈中的本地变量表)中引用的对象
b.方法区中静态属性引用的对象
c.方法区中常量引用的对象
d.本地方法栈中JNI引用的对象
2.不可达对象一定会被回收吗不是。
执行垃圾回收前JVM会执行不可达对象的finalize方法,如果执行完毕之后该对象变为可达,则不会被回收它。
但一个对象的finalize方法只会被执行一次。参考资料:《深入理解Java虚拟机》周志明

你可能感兴趣的:(java一些小知识点)