8

正如 Hans Boehm 在 Google I/O '17 演讲“如何在 Android 中管理本机 C++ 内存”中建议我使用PhantomReference该类来确保正确删除本机对等点。

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

虽然这种方法效果很好,但它不能扩展。我将需要创建相当多的对象,但我还没有找到一种方法来创建一个基类(对于我的对象或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);
    }
  }
}
4

1 回答 1

8

您可能会查看 Java 9 的CleanerAPI,它解决了类似的任务,围绕 a 构建清理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()显式地触发清理,但它是线程安全的,因此创建一个后台后台线程来清理项目是没有问题的,就像在原始项目中一样。

将操作表示为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-resourcesclose()构造一起使用,但如果没有简单的块作用域,也可以手动调用。手动关闭它的好处,不仅是底层资源更早关闭,而且幻象对象也被移除Set并且永远不会入队,使整个生命周期更有效率,特别是当你创建和使用大量对象时短期内。但是如果close()没有被调用,清洁器最终会被垃圾收集器排队。

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

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

于 2017-12-15T10:24:28.647 回答