虚引用与堆外内存研究(基于DirectByteBuffer源码)

(1)堆内存
ByteBuffer buffer = ByteBuffer.allocate(1024);
字节数组保存数据HeapByteBuffer
   
(2)堆外内存
ByteBuffer directByteBuffer = ByteBuffer.allocateDirect(1024);
Buffer类有个:long address;用于保存堆外空间地址
unsafe来分配以及最后回收空间
构建Cleaner对象,继承虚引用,将当前堆外内存以及垃圾清理线程对象传递过去,GC发生后,调用Deallocator的clean方法,内部有调用unsafe来回收堆外内存。

DirectByteBuffer(int cap) {              
        //** address,base分配堆外内存空间之后的起始地址
        long base = 0;
        try {
            base = unsafe.allocateMemory(size);
        } catch (OutOfMemoryError x) {
            Bits.unreserveMemory(size, cap);
            throw x;
        }
        unsafe.setMemory(base, size, (byte) 0);
        if (pa && (base % ps != 0)) {
            // Round up to page boundary
            address = base + ps - (base & (ps - 1));
        } else {
            address = base;
        }
        //构建Cleaner对象用于跟踪DirectByteBuffer对象的垃圾回收,以实现当DirectByteBuffer被垃圾回收时,堆外内存也会被释放
        cleaner = Cleaner.create(this, new Deallocator(base, size, cap));
        //********************************
}
/**
**继承虚引用
* clener回收列表:其中first是Cleaner类的静态变量,Cleaner对象在初始化时会被添加到Clener链表中, * 和first形成引用关系,。
* ReferenceQueue是用来保存需要回收的Cleaner对象。
* cleaner对象唯一保存了堆外内存的数据(开始地址、大小和容量)
* 若DerectByteBuffer对象被回收,那么把它内部的Cleaner对象放入到ReferenceQueue中,并触发clean * 方法。释放堆外内存。
*/
public class Cleaner extends PhantomReference<Object> {
    private static final ReferenceQueue<Object> dummyQueue = new ReferenceQueue();
    private static Cleaner first = null;
    private Cleaner next = null;
    private Cleaner prev = null;
    private final Runnable thunk;
    
    // var1 即directBuffer, var2即runnable对象Deallocator对象
    private Cleaner(Object var1, Runnable var2) {
        super(var1, dummyQueue);
        this.thunk = var2;
    }
    public static Cleaner create(Object var0, Runnable var1) {
        return var1 == null ? null : add(new Cleaner(var0, var1));
    }
    public void clean() {
        //清理
        if (remove(this)) {
            try {
                this.thunk.run();
            } catch (final Throwable var2) {
             ***   
            }
        }
    }
}
// 回收线程,保存堆外内存信息以及执行回收操作
private static class Deallocator implements Runnable{
        private static Unsafe unsafe = Unsafe.getUnsafe();
        private long address;
        private long size;
        private int capacity;
        private Deallocator(long address, long size, int capacity) {
            assert (address != 0);
            this.address = address;
            this.size = size;
            this.capacity = capacity;
        }
        public void run() {
            if (address == 0) {
                return;
            }
            unsafe.freeMemory(address);
            address = 0;
            Bits.unreserveMemory(size, capacity);
        }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值