Java 垃圾回收和对象引用

一、垃圾回收

Java中垃圾回收算法是可达性分析,要真正回收一个对象,会经历两次标记过程,如果对象在通过可达性算法分析后,没有发现和GC Root相连的引用链,那么就会进行第一次标记(修改Mark Word),标记完之后会进行筛选,判定是否需要执行finalize()方法,如果不需要则直接回收,如果需要,会加入F-Queue队列中,等待执行完finalize之后回收。

下面是一个例子

public class FinalizeGc {

    static  FinalizeGc fg = null;
    public static  void main(String[] args){
        fg = new FinalizeGc();
        fg = null;
        System.gc();
        try {
            Thread.sleep(500);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.print("end call by "+Thread.currentThread());
    }
    @Override
    protected void finalize() throws Throwable {
        super.finalize();
        System.out.println("finalize is called by "+Thread.currentThread());
    }
}

输出结果

finalize is called by Thread[Finalizer,8,system]
end call by Thread[main,5,main]

注意:

①如果对象在finalize方法中被其他外部变量重新引用,对象将不会被回收,自行设计实验即可。

② finalize被调用在Finalizer线程,该线程由虚拟机创建,属于deamon线程,而且优先级很低,因此未必等待该方法执行完就会回收垃圾,有可能提取回收。

二、对象引用

1、关于四种引用关系

我们知道对象引用主要分为4种,强引用、软引用、弱引用、虚引用。强引用是编程时主要的引用方式,软引用和弱引用主要解决内存不足的问题,此外弱引用还有解决对方引用不能正确释放的问题。虚引用相当特殊,主要用来监控对象的回收。

1.强引用:

当这个对象被强引用时,就算出现空间内存不足,也要出现oom的情况

2.软引用:

当这个对象被软引用时,一般垃圾回收器不会管它,但是出现空间内存不足,就要被回收。

3.弱引用:

当这个对象被弱引用时,一般还没有出现空间内存不足的情况时,如果垃圾回收器线程扫描到它,就会被回收。

4.虚引用:

当这个对象仅仅被虚引用时,该对象可以在任何时候被垃圾回收。

虚引用主要是用来跟踪对象被垃圾回收的活动。

虚引用与软引用和弱引用的一个区别在于:虚引用必须和引用队列(ReferenceQueue)联合使用。

当一个对象被回收时,如果发现还有一个虚引用,在回收该对象前,就要把这个虚引用加入到与之相关的引用队列中。

程序通过了解该引用队列,可以判断该对象是否被回收。相当于日志追踪。

2、关于弱引用和虚引用

测试时我们发现,弱引用和虚引用有很多相似之处,也是同样在内存不足是释放对象

测试代码如下:

        Object obj = new long[1024];
        Reference<Object> wf = new PhantomReference<>(obj,new ReferenceQueue<Object>());
        List<Object> eatMemorylist = new ArrayList<>();
        obj  = null;
        while (!wf.isEnqueued()) {
            eatMemorylist.add(new byte[5*1024*1024]);
            //有时候会返回null
            System.out.println(" =>> Object: " + wf.get() + " isEnqueued:" + wf.isEnqueued());
            // wf.isEnQueued();//返回是否被垃圾回收器标记为即将回收的垃圾
            Thread.sleep(1000);

        }

那么问题来了,弱引用和虚引用是否可互相代替呢?

答案:虚引用的get方法实现不允许你获取引用对象


public class PhantomReference<T> extends Reference<T> {

    /**
     * Returns this reference object's referent.  Because the referent of a
     * phantom reference is always inaccessible, this method always returns
     * <code>null</code>.
     *
     * @return  <code>null</code>
     */
    public T get() {
        return null;  //无法使用被引用的对象
    }

    /**
     * Creates a new phantom reference that refers to the given object and
     * is registered with the given queue.
     *
     * <p> It is possible to create a phantom reference with a <tt>null</tt>
     * queue, but such a reference is completely useless: Its <tt>get</tt>
     * method will always return null and, since it does not have a queue, it
     * will never be enqueued.
     *
     * @param referent the object the new phantom reference will refer to
     * @param q the queue with which the reference is to be registered,
     *          or <tt>null</tt> if registration is not required
     */
    public PhantomReference(T referent, ReferenceQueue<? super T> q) {
        super(referent, q);
    }

}

三、引用Size
Shallow Size
对象自身占用的内存大小,不包括它引用的对象。
针对非数组类型的对象,它的大小就是对象与它所有的成员变量大小的总和。当然这里面还会包括一些java语言特性的数据存储单元。
针对数组类型的对象,它的大小是数组元素对象的大小总和。

Retained Size
Retained Size=当前对象大小+当前对象可直接或间接引用到的对象的大小总和。(间接引用的含义:A->B->C, C就是间接引用)
换句话说,Retained Size就是当前对象被GC后,从Heap上总共能释放掉的内存。
不过,释放的时候还要排除被GC Roots直接或间接引用的对象。他们暂时不会被被当做Garbage。

看图理解Retained Size

上图中,GC Roots直接引用了A和B两个对象。

A对象的Retained Size=A对象的Shallow Size
B对象的Retained Size=B对象的Shallow Size + C对象的Shallow Size

这里不包括D对象,因为D对象被GC Roots直接引用。
如果GC Roots不引用D对象呢?

此时,
B对象的Retained Size=B对象的Shallow Size + C对象的Shallow Size + D对象的Shallow Size

参考: https://www.cnblogs.com/0616--ataozhijia/p/3694924.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值