JAVA 强、软、弱、虚快速理解区别

2022-05-20  本文已影响0人  一枚路过的程序猿

JAVA中类型引用分为:强引用、软引用、弱引用、虚引用

强引用

强引用,GC默认不回收,只有在对象没有被引用的时候进行GC回收

/**
 * @Description: 强引用:对象没有引用GC开始回收
 * @Author haodd
 * @Date 2022/5/19 11:08
 * @Version 1.0
 */
public class StrongRef {

    public static void main(String[] args) throws InterruptedException {

        StrongRef m = new StrongRef();

        System.out.println(m);

        System.gc();

        System.out.println(m);

        m = null;

        System.gc();

        /** 此处等待,防止GC线程没有执行完 */
        Thread.sleep(100);

        System.out.println(m);

    }

    @Override
    protected void finalize() throws Throwable {
        System.out.println("GC垃圾回收");
        super.finalize();
    }
}

软引用

软引用,GC默认不回收,在堆内存不够的时候,GC回收软应用
此处进行实验需要设置JVM最大堆内存,-Xmx20M

import java.lang.ref.SoftReference;

/**
 * @Description: 软引用:默认GC不回收,如果堆内存满了,软引用被GC回收
 * @Author haodd
 * @Date 2022/5/19 11:14
 * @Version 1.0
 */
public class SoftRef {

    //TODO 实验环境的时候需要设置JVM最大堆内存:-Xmx20M
    public static void main(String[] args) throws InterruptedException {

        SoftReference<byte[]> m = new SoftReference<>(new byte[1024 * 1024 * 10]);

        System.out.println(m.get());

        System.gc();

        Thread.sleep(800);

        System.out.println(m.get());

        byte[] bytes = new byte[1024 * 1024 * 11];

        System.out.println(m.get());

    }

}

软引用

弱引用,遇到GC就回收

import java.lang.ref.WeakReference;

/**
 * @Description: 弱引用,遇到GC就会被回收
 * @Author haodd
 * @Date 2022/5/19 14:02
 * @Version 1.0
 */
public class WeakRef {

    public static void main(String[] args) throws InterruptedException {

        WeakReference<WeakRef> m = new WeakReference<>(new WeakRef());

        System.out.println(m.get());

        System.gc();

        Thread.sleep(500);

        System.out.println(m.get());

    }

    /** GC垃圾回收的会后会调用此方法,但不一定每次都会调用 */
    @Override
    protected void finalize() throws Throwable {
        System.out.println("GC垃圾回收");
        super.finalize();
    }
}

虚引用

虚引用,创建虚引用的时候,需要传入对象和queue,当虚引用的对象被回收后,会在queue中添加被回收的记录
虚引用可以用来管理堆外内存,正常网络通信下,数据先到电脑内存,最后才能到JVM内存,经过两次拷贝,效率低
所以JVM提供了直接内存管理,因为此处直接操作的是操作系统内存,所以JVM没办法管理堆外内存,因此在会在被回收的时候才会给queue中发送一条消息,告诉你该清理堆外内存了

import java.lang.ref.PhantomReference;
import java.lang.ref.ReferenceQueue;
import java.util.concurrent.TimeUnit;

/**
 * @Description: 虚引用,创建虚引用的时候,需要传入对象和queue,当虚引用的对象被回收后,会在queue中添加被回收的记录
 *               虚引用可以用来管理堆外内存,正常网络通信下,数据先到电脑内存,最后才能到JVM内存,经过两次拷贝,效率低
 *               所以JVM提供了直接内存管理,因为此处直接操作的是操作系统内存,
 *               所以JVM没办法管理堆外内存,因此在会在被回收的时候才会给queue中发送一条消息,告诉你该清理堆外内存了
 * @Author haodd
 * @Date 2022/5/19 17:57
 * @Version 1.0
 */
public class PhantomRef {

    public static void main(String[] args) throws InterruptedException {

        Object obj = new Object();
        ReferenceQueue<Object> QUEUE = new ReferenceQueue<>();
        PhantomReference<Object> reference = new PhantomReference<>(obj,QUEUE);

        obj = null;
        System.out.println(reference.get());

        System.gc();
        TimeUnit.SECONDS.sleep(6);

        System.out.println(reference.get());
        System.out.println(QUEUE.poll());

    }

}

上一篇下一篇

猜你喜欢

热点阅读