强、软、弱、虚,你是哪一种?

简介: 强、软、弱、虚,你是哪一种?

前言

了不起最近在整理一些面试资料,发现对于强、软、弱、虚引用的资料少之又少,所以决定整理一下关于这方面的资料,方便金三银四跳槽。

强引用

特点

强引用是较为普遍的一种引用,在我们编写Java代码时,绝大多数的引用,使用的都是强引用。当一个对象使用的是强引用时,JVM垃圾回收器不会回收该对象,即使内存不足,JVM宁愿抛出OOM异常,也不会主动去回收该对象。

代码示例

public class StrongReference {
    public static void main(String[] args) {
        MyObject myObject = new MyObject();
        System.gc();
        System.out.println("第一次gc,myObject = " + myObject);
        // 手动置为Null,并通知JVM进行垃圾回收
        myObject = null;
        System.gc();
        System.out.println("第二次gc,myObject = " + myObject);
    }
}
class MyObject{
    @Override
    protected void finalize() throws Throwable {
        System.out.println("execute finalize method...");
    }
}
// 执行结果:
// 第一次gc,myObject = ref.MyObject@1b6d3586
// 第二次gc,myObject = null
// execute finalize method...

软引用

特点

软引用是一种较强的引用类型,当内存足够时,JVM不会回收软引用的对象,当内存不足时,JVM会回收软引用的对象。

代码示例

前置条件:调整JVM参数,将最大内存调整为10MB。-Xms10m -Xmx10m
import java.lang.ref.SoftReference;
import java.util.concurrent.TimeUnit;
public class SoftReferenceExample {
    public static void main(String[] args) {
        SoftReference softReference = new SoftReference<MyObject>(new MyObject());
        System.out.println("gc before, softReference = " + softReference.get());
        System.gc();
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("gc after, softReference = " + softReference.get());
        try {
            // 创建20MB的字节数组
            byte[] bytes = new byte[20 * 1024 * 1024];
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            System.out.println("memory not enough, softReference = " + softReference.get());
        }
    }
}
// 执行结果:
// gc before, softReference = ref.MyObject@1b6d3586
// gc after, softReference = ref.MyObject@1b6d3586
// Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
//   at ref.SoftReferenceExample.main(SoftReferenceExample.java:22)
// memory not enough, softReference = null
// execute finalize method...

弱引用

特点

弱引用是一种比软引用更弱的引用类型,无论内存是否充足,JVM垃圾回收器都会立即回收弱引用对象。

代码示例

import java.lang.ref.WeakReference;
import java.util.concurrent.TimeUnit;
public class WeakReferenceExample {
    public static void main(String[] args) {
        WeakReference weakReference = new WeakReference<>(new MyObject());
        System.out.println("gc before, softReference = " + weakReference.get());
        System.gc();
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("gc after, softReference = " + weakReference.get());
    }
}
// 执行结果:
// gc before, softReference = ref.MyObject@1b6d3586
// execute finalize method...
// gc after, softReference = null

虚引用

特点

虚引用是最弱的引用类型,主要用于跟踪对象是否被垃圾回收器回收。同时,虚引用必须与一个引用队列一同使用。当JVM准备回收一个对象时,如果发现它是虚引用,那么会将这个虚引用加入到与之关联的一个引用队列中。

代码示例

import java.lang.ref.PhantomReference;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.TimeUnit;
public class PhamtonReferenceExample {
    public static void main(String[] args) throws InterruptedException {
        ReferenceQueue<MyObject> referenceQueue = new ReferenceQueue<>();
        PhantomReference<MyObject> phantomReference = new PhantomReference<>(new MyObject(), referenceQueue);
        List<byte[]> list = new ArrayList<>();
        new Thread(() -> {
            while (true) {
                list.add(new byte[2 * 1024 * 1024]);
                try {
                    TimeUnit.SECONDS.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(phantomReference.get() + "\t" + "list add success!!!");
            }
        }).start();
        new Thread(() -> {
            while (true) {
                Reference<? extends MyObject> ref = referenceQueue.poll();
                if (ref != null) {
                    System.out.println("虚对象回收加入了对列");
                    break;
                }
            }
        }).start();
    }
}
// 执行结果:
// execute finalize method...
// null  list add success!!!
// null  list add success!!!
// null  list add success!!!
// Exception in thread "Thread-0" java.lang.OutOfMemoryError: Java heap space
//    at ref.PhamtonReferenceExample.lambda$main$0(PhamtonReferenceExample.java:21)
//   at ref.PhamtonReferenceExample$$Lambda$1/1324119927.run(Unknown Source)
//    at java.lang.Thread.run(Thread.java:748)
// 虚对象回收加入了对列

总结

引用类型可用于灵活管理对象的生命周期,且在Java源码中有较多的应用场景,如ThreadLocal使用的就是弱引用。希望本文能够帮助小伙伴们对Java的四种引用类型有更深入的了解。

相关文章
|
存储 缓存 Oracle
常识四堆外内存
常识系列,作为一名互联网门外汉的科普系列 堆外内存除了在像netty开源框架中,在平常项目中使用的比较少,在现前的项目中,QPS要求高的系统中,堆外内存作为其中一级缓存是相当有成效的。所以来学习一下,文中主要涉及到这三分部内容 1. 堆外内存是什么?与堆内内存的区别 2. 怎么分配,与GC的影响 3. 开源框架使用 这篇文章写到最后,发现还只是回答了开源框架OHC的Why not use ByteBuffer.allocateDirect()?
1352 1
常识四堆外内存
|
5月前
|
存储 Swift
Swift开发——弱占用
Swift的自动引用计数(ARC)管理类实例内存,通过强引用保持实例存活。当出现强引用循环时,可使用`weak`关键字创建弱引用,避免阻止实例释放。弱引用在不再被强引用时导致对象立即释放。示例中,添加`weak`至`author`和`book`变量防止引用循环,使得两者析构器均执行,释放内存。图2展示了弱引用结构,当解除所有强引用后,ARC自动释放实例,调用析构器。
208 1
Swift开发——弱占用
|
6月前
|
机器学习/深度学习 编解码
ESRGAN超分辨网络
ESRGAN超分辨网络
80 0
|
6月前
|
机器学习/深度学习 传感器 编解码
BSRGAN超分辨网络
BSRGAN超分辨网络
85 0
|
运维 网络协议
【每日一记】OSPF区域划分详讲、划分区域的优点好处
【每日一记】OSPF区域划分详讲、划分区域的优点好处
306 0
|
算法 调度
转:使用匈牙利算法对局域网共享软件有哪些好处
在局域网共享软件中,匈牙利算法主要应用于解决资源分配的问题。局域网共享软件可能存在多个用户同时访问同一文件或打印机的情况,为了确保资源的公平共享,需要对资源进行分配。
89 2
|
负载均衡 网络协议 网络虚拟化
变形金刚外传0x08-动态路由实现物理-逻辑互访
在上一篇分享中,我通过设置静态路由条目,实现了NSX-T演示环境中,逻辑网络与物理网络的三层互通。虽然静态路由没有动态路由协议的灵活性,但在一些情况下,静态路由却是首选的方案:
|
编解码
失真的概念和定义
失真的概念和定义
229 0
|
机器学习/深度学习 自然语言处理
注意力机制可分为软和硬两类,空间域,通道域
注意力机制是当前深度学习领域比较流行的一个概念。其模仿人的视觉注意力模式,每次只关注与当前任务最相关的源域信息,使得信息的索取更为高效。
717 0
注意力机制可分为软和硬两类,空间域,通道域
|
Java
强、软、弱、虚引用
强、软、弱、虚引用
91 0