java中的强软弱虚_java中的引用类型之强软弱虚详解

前言

java中的引用类型共4种:强软弱虚,具体每种类型的特点和应用场景。记录下。本文是看了马士兵老师的视频后记录整理的。加深印象。

基本概念

1. 强引用

强引用是使用最普遍的引用。如果一个对象具有强引用,那垃圾回收器绝不会回收它。当内存空间不足时,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不会靠随意回收具有强引用的对象来解决内存不足的问题。

c4d7b278d26e4e7ef5571b3dcefee871.png

显式地设置M对象为null,或让其超出对象的生命周期范围,则gc认为该对象不存在引用,这时就可以回收这个对象

示例代码

声明一个M类

public class M {

/**

* 当这个对象会被回收的时候,finalize会被调用

*

* @throws Throwable

*/

@Override

protected void finalize() throws Throwable {

System.out.println("finalize");

}

}

强引用调用

public class NormalReference {

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

M m = new M();

m = null;

System.gc();

// 如果不写 main方法退出。System.gc()在垃圾回收线程里;

// 有可能还没来得及回收main方法就退出了

System.in.read();

}

}

输出结果

finalize

2. 软引用

软引用对象是在jvm内存不够的时候才会被回收

5f454d5df1755de62fc905138a12263f.png

代码示例

public class Soft {

public static void main(String[] args) {

System.out.println(Runtime.getRuntime().totalMemory() / 1024 / 1024);

//10m

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

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

System.gc();

try {

Thread.sleep(1500);

} catch (InterruptedException e) {

e.printStackTrace();

}

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

// 再分配一个数组,heap装不下去,这时候系统会垃圾回收,

// 先回收一次,如果不够,会把软引用回收

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

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

}

}

设置程序运行参数: -Xmx20M

运行结果

19

[B@1540e19d

[B@1540e19d

null

我们可以看到,这个时候已经被回收了。

应用场景:软引用时候做缓存

3. 弱引用

弱引用与软引用的区别在于:只具有弱引用的对象拥有更短暂的生命周期。在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存

代码示例

public class Weak {

public static void main(String[] args) {

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

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

System.gc();

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

}

}

执行结果

M@1540e19d

null

finalize

4. 虚引用

如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。

虚引用必须和引用队列(ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。

d74cc473d492bf7bb04e10572beb7dcb.png

示例

public class Phantom {

private static final List LIST = new LinkedList<>();

private static final ReferenceQueue QUEUE = new ReferenceQueue<>();

public static void main(String[] args) {

PhantomReference phantomReference = new PhantomReference<>(new M(), QUEUE);

new Thread(

() -> {

while (true) {

LIST.add(new byte[1024 * 1024]);

try {

Thread.sleep(1000);

} catch (InterruptedException e) {

Thread.currentThread().interrupt();

}

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

}

}

).start();

new Thread(() -> {

while (true) {

Reference extends M> poll = QUEUE.poll();

if (poll != null) {

System.out.println("-------虚引用对象被jvm回收了------" + poll);

}

}

}).start();

}

}

运行结果

finalize

null

Exception in thread "Thread-0" java.lang.OutOfMemoryError: Java heap space

at Phantom.lambda$main$0(Phantom.java:18)

at Phantom$$Lambda$1/1078694789.run(Unknown Source)

at java.lang.Thread.run(Thread.java:748)

-------虚引用对象被jvm回收了------java.lang.ref.PhantomReference@688ee48d

应用场景:堆外内存的管理

总结

到此这篇关于java中引用类型之强软弱虚的文章就介绍到这了,更多相关java引用类型之强软弱虚内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值