使用常规PhantomReference类

时间:2017-09-10 19:06:18

标签: java android memory-management phantom-reference

Hans Boehm在Google I / O 17" How to Manage Native C++ Memory in Android"建议我使用PhantomReference类来确保正确删除本地对等。

18 min 57 sec的链接视频中,他展示了一个对象的示例实现,该对象将其自身注册到PhantomReference类的类型。这个PhantomReference课程,然后在19 min 49 sec显示。所以我为我的示例对象复制了他的方法。见下文。

虽然这种方法工作正常,但它不能扩展。我需要创建相当多的对象,但我还没有找到创建基类的方法(对于我的对象或PhantomReference基类),它将接受任何对象并正确处理本机删除。

如何创建一个可以在提供的对象上调用本机静态方法的通用基类PhantomReference类?

我尝试转换PhantomReference泛型,但原生静态删除方法阻碍了实现。

我的WorkViewModel

import android.databinding.*;

public class WorkViewModel extends BaseObservable
{
  private long _nativeHandle;

  public WorkViewModel(Database database, int workId)
  {
    _nativeHandle = create(database.getNativeHandle(), workId);
    WorkViewModelPhantomReference.register(this, _nativeHandle);
  }

  private static native long create(long databaseHandle, int workId);
  static native void delete(long nativeHandle);

  @Bindable
  public native int getWorkId();
  public native void setWorkId(int workId);
}

我的WorkViewModelPhantomReference

import java.lang.ref.*;
import java.util.*;

public class WorkViewModelPhantomReference extends PhantomReference<WorkViewModel>
{
  private static Set<WorkViewModelPhantomReference> phantomReferences = new HashSet<WorkViewModelPhantomReference>();
  private static ReferenceQueue<WorkViewModel> garbageCollectedObjectsQueue = new ReferenceQueue<WorkViewModel>();
  private long _nativeHandle;

  private WorkViewModelPhantomReference(WorkViewModel workViewModel, long nativeHandle)
  {
    super(workViewModel, garbageCollectedObjectsQueue);
    _nativeHandle = nativeHandle;
  }

  public static void register(WorkViewModel workViewModel, long nativeHandle)
  {
    phantomReferences.add(new WorkViewModelPhantomReference(workViewModel, nativeHandle));
  }

  public static void deleteOrphanedNativePeerObjects()
  {
    WorkViewModelPhantomReference reference;

    while((reference = (WorkViewModelPhantomReference)garbageCollectedObjectsQueue.poll()) != null)
    {
      WorkViewModel.delete(reference._nativeHandle);
      phantomReferences.remove(reference);
    }
  }
}

1 个答案:

答案 0 :(得分:5)

您可以查看Java 9的Cleaner API,它可以解决类似的任务,围绕PhantomReference进行清理,并实现类似的功能,并根据您的需要进行调整。由于您不需要支持多个清洁工,因此您可以使用static注册方法。我建议保留引用的抽象,即Cleanable接口,以确保不能调用任何继承的引用方法,尤其是当clear()clean()容易混淆时:

public class Cleaner {
    public interface Cleanable {
        void clean();
    }
    public static Cleanable register(Object o, Runnable r) {
        CleanerReference c = new CleanerReference(
                Objects.requireNonNull(o), Objects.requireNonNull(r));
        phantomReferences.add(c);
        return c;
    }
    private static final Set<CleanerReference> phantomReferences
                                             = ConcurrentHashMap.newKeySet();
    private static final ReferenceQueue<Object> garbageCollectedObjectsQueue
                                              = new ReferenceQueue<>();

    static final class CleanerReference extends PhantomReference<Object>
                                        implements Cleanable {
        private final Runnable cleaningAction;

        CleanerReference(Object referent, Runnable action) {
            super(referent, garbageCollectedObjectsQueue);
            cleaningAction = action;
        }
        public void clean() {
            if(phantomReferences.remove(this)) {
                super.clear();
                cleaningAction.run();
            }
        }
    }
    public static void deleteOrphanedNativePeerObjects() {
        CleanerReference reference;
        while((reference=(CleanerReference)garbageCollectedObjectsQueue.poll()) != null) {
            reference.clean();
        }
    }
}

这使用Java 8功能;如果ConcurrentHashMap.newKeySet()不可用,您可以改为使用Collections.newSetFromMap(new ConcurrentHashMap<CleanerReference,Boolean>())

它保持deleteOrphanedNativePeerObjects()显式触发清理,但它是线程安全的,因此创建一个守护进程后台线程来清除项目就像它们一样排队就没有问题了,就像原来一样。 / p>

将操作表示为Runnable允许将此操作用于任意资源,并且使Cleanable返回允许支持显式清理而不依赖于垃圾收集器,同时仍然具有那些对象的安全网尚未关闭。

public class WorkViewModel extends BaseObservable implements AutoCloseable
{
    private long _nativeHandle;
    Cleaner.Cleanable cleanable;

    public WorkViewModel(Database database, int workId)
    {
      _nativeHandle = create(database.getNativeHandle(), workId);
      cleanable = createCleanable(this, _nativeHandle);
    }
    private static Cleaner.Cleanable createCleanable(Object o, long _nativeHandle) {
        return Cleaner.register(o, () -> delete(_nativeHandle));
    }

    @Override
    public void close() {
        cleanable.clean();
    }

    private static native long create(long databaseHandle, int workId);
    static native void delete(long nativeHandle);

    @Bindable
    public native int getWorkId();
    public native void setWorkId(int workId);

}

通过实现AutoCloseable,它可以与try-with-resources构造一起使用,但如果没有简单的块范围,也可以手动调用close()。手动关闭它的好处不仅在于底层资源更早关闭,幻像对象也从Set中删除而且永远不会排队,从而使整个生命周期更加高效,尤其是在创建时并在短期内使用很多对象。但是如果没有调用close(),清理器最终会被垃圾收集器排队。

对于支持手动关闭的类,保留一个标志会很有用,可以在关闭后检测并拒绝使用它的尝试。

如果lambda表达式不适用于您的目标,您可以通过内部类实现Runnable;它仍然比创建幻像引用的另一个子类更简单。必须注意不要捕获this实例,这就是为什么在上面的示例中将创建移动到static方法的原因。如果范围不this,则无法意外捕获。该方法还将referent声明为Object,以强制使用参数值而不是实例字段。