PhantomReference and memory mangement

软引用和弱引用可以单独使用,但虚引用不能单独使用,单独使用虚引用没有太大的意义。虚引用的主要作用就是跟踪对象被垃圾回收的状态,程序可以通过检查与虚引用关联的引用队列中是否已经包含指定的虚引用,从而了解虚引用所引用对象是否即将被回收。

引用队列由Java.lang.ref.ReferenceQueue类表示,它用于保存被回收后对象的引用。当把软引用,弱引用和引用队列联合使用时,系统回收被引用的对象之后,将会把被回收对象对应的引用添加到关联的引用队列中。与软引用和弱引用不同的是,虚引用在对象被释放之前,将把对应的虚引用添加到它的关联的引用队列中,这使得可以在对象被回收之前采取行动。

虚引用通过PhantomReference类实现,它完全类似于没有引用。虚引用对对象本身没有太大影响,对象甚至感觉不到虚引用的存在。如果一个对象只有一个虚引用,那它和没有引用的效果大致相同。

import java.lang.ref.PhantomReference;

import java.lang.ref.ReferenceQueue;

 

 

public class PhantomReferenceTest {

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

      Stringstr = new String("Java");

      ReferenceQueue<String>rq = newReferenceQueue<String>();

      PhantomReference<String>pr = newPhantomReference<String>(str,rq);

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

      str = null;

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

      System.gc();

      System.runFinalization();

      System.out.println(rq.poll());

      System.out.println(rq.poll()==pr);

    }

}

Output:

null

null

java.lang.ref.PhantomReference@61de33

false

 

使用这些引用类可以避免在程序只习惯期间将对象留在内存中。如果以软引用,弱引用或虚引用的方法引用对象,垃圾回收器就能够随意地释放对象。如果希望尽可能减小程序在其生命周期中所占用的内存大小,这些引用类就很有好处。最后需要指出的是:要使用这些特殊的引用类,就不能保留对对象的强引用,如果保留了就会浪费这些类所提供的任何好处。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
PhantomReference(幽灵引用)是 Java 的一种引用类型,它是弱引用(WeakReference)的一种变种。与弱引用不同的是,当一个对象只剩下幽灵引用时,它在下一次垃圾回收时就会被回收,并且在回收之前,幽灵引用对象会被放入一个队列,以便进行一些处理操作。 PhantomReference 可以用于实现一些特殊的功能,例如在对象被回收时进行一些资源释放或清理操作。通常,我们可以通过继承 PhantomReference 类或使用它的包装类 PhantomReference<T> 来创建幽灵引用对象。 下面是一个使用幽灵引用的示例代码: ``` import java.lang.ref.PhantomReference; import java.lang.ref.ReferenceQueue; public class ResourceCleaner extends Thread { private ReferenceQueue<Resource> queue; public ResourceCleaner() { this.queue = new ReferenceQueue<>(); setDaemon(true); start(); } public void register(Resource resource) { PhantomReference<Resource> ref = new PhantomReference<>(resource, queue); // 注册幽灵引用对象 } @Override public void run() { try { while (true) { PhantomReference<Resource> ref = (PhantomReference<Resource>) queue.remove(); // 从队列取出幽灵引用对象 Resource resource = ref.get(); if (resource != null) { // 执行资源释放或清理操作 resource.release(); } ref.clear(); } } catch (InterruptedException ex) { // 终止线程 } } } ``` 在上述代码,我们创建了一个 ResourceCleaner 类,它继承自 Thread 类,并重写了其 run() 方法。在 ResourceCleaner 类,我们定义了一个 ReferenceQueue<Resource> 对象,用于存放幽灵引用对象。在 register() 方法,我们创建了一个幽灵引用对象,并将其注册到队列。当对象被回收时,幽灵引用对象就会被放入队列,从而触发 ResourceCleaner 线程的执行。在 ResourceCleaner 线程,我们可以从队列取出幽灵引用对象,并执行资源释放或清理操作。需要注意的是,在执行完资源释放或清理操作之后,我们还需要调用幽灵引用对象的 clear() 方法来清除它,以便让 JVM 可以正常回收幽灵引用对象本身。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值