强软弱虚引用

强软弱虚引用

强引用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
/**
* @author zc
* @date 2021/1/15 10:39
* 强引用是使用最普遍的引用。如果一个对象具有强引用,那垃圾回收器绝不会回收它。当内存空间不足,Java虚拟机宁愿抛出OOM,也不会靠随意回收具有强引用的对象来解决内存不足的问题。
*/
public class QiangReference {

public static void main(String[] args) {
T t = new T();
System.out.println(t);
t = null;
System.out.println(t);
System.gc();
System.out.println(t);
}
}

软引用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
/**
* 软引用
* 在一个SoftReference有一个对象,该对象占用堆内存,当堆内存容量不够时,回收它。
* <p>
* 设置JVM堆内存20M:-Xmx20M
* <p>
* 如果一个对象只具有软引用,则内存空间足够,垃圾回收器就不会回收它;如果内存空间不足了,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就可以被程序使用。
* 软引用可用来实现内存敏感的高速缓存。
* <p>
* 应用场景:缓存,用时缓存起来,当分配给其他对象内存不够使,回收它
*
* @author zc
* @date 2021/1/15 10:44
*/
public class RuanReference {


public static void main(String[] args) throws InterruptedException {
//new一个软应用对象,里面存放10M byte数据
SoftReference<byte[]> sr = new SoftReference<>(new byte[1024 * 1024 * 10]);
//获取它
System.out.println(sr.get());
//GC回收
System.gc();
Thread.sleep(1000);
//再次获取
System.out.println(sr.get());
//new一个13M byte对象,
byte[] b = new byte[1024 * 1024 * 13];
//此时发现被回收了
System.out.println(sr.get());
}

}

弱引用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
/**
* 弱引用
* 一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。
* 软引用和弱引用都可以和一个引用队列(ReferenceQueue)联合使用,如果所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用或软引用加入到与之关联的引用队列中。
* ThreadLocal类的静态内部类ThreadLocalMap的静态内部类Entry中的k,就是弱引用
* <p>
* 应用场景:ThreadLocal,@Transactional就是用ThreadLocal实现的
*
* @author zc
* @date 2021/1/15 13:12
*/
public class RuoReference {

public static void main(String[] args) throws InterruptedException {
//new一个软应用对象,里面存放10M byte数据
WeakReference<byte[]> sr = new WeakReference<>(new byte[1024 * 1024 * 10]);
//获取它
System.out.println(sr.get());
//GC回收
System.gc();
//再次获取,发现被回收了
System.out.println(sr.get());
}
}

虚引用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
/**
* 虚引用
* <p>
* 虚引用”顾名思义,就是形同虚设,与其他几种引用都不同,虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。
* 虚引用主要用来跟踪对象被垃圾回收器回收的活动,且虚引用必须和引用队列(ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。
* <p>
* 程序可以通过判断引用队列中是否已经加入了虚引用,来了解被引用的对象是否将要被垃圾回收。如果程序发现某个虚引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之前采取必要的行动。
* <p>
* 应用场景:主要用来管理堆外内存。NIO,Netty底层实现
*
* @author zc
* @date 2021/1/15 13:20
*/
public class XuReference {
public static void main(String[] args) {
T t = new T();
ReferenceQueue queue = new ReferenceQueue();
PhantomReference r = new PhantomReference(t, queue);
//获取它
System.out.println(r.get());
//GC回收
System.gc();
//再次获取,发现被回收了
System.out.println(r.get());
}
}

强软弱虚引用
https://happyloves.cn/20220831/1ceca005f04d.html
作者
赵小胖
发布于
2022年8月31日
许可协议