7

是否可以在 Java 中创建一个弱引用,其对象只有在指定条件返回时才能发送到垃圾收集器true

假设我有一个类似于缓存的东西,它将 ID 号映射到一些数据:

Map<Integer, SomeData> cache = new HashMap<>();

SomeData有两个重要的方法 - void updateTime(),它只是将一个内部变量设置为当前时间,以及boolean canBeDeleted(),它检查对象是否在过去 10 分钟内被使用过(只需比较当前时间和保存的时间加上 10 分钟)。如果在这段时间内没有使用它,则该方法返回true并且可以从缓存中删除该对象...

但是,当我使用弱引用而不是强引用创建缓存时:

Map<Integer, WeakReference<SomeData>> cache = new HashMap<>();

然后唯一WeakReference检查的可能是对该对象的强引用,但我想它是否也检查了我的条件canBeDeleted()并且如果它返回则不删除引用false。有没有办法做到这一点?

(最重要的是,存在内存泄漏,我不知道如何解决它......当 WeakReference 中的对象被删除时,映射仍然包含不必要的键/值对)。

提前致谢。

4

2 回答 2

2

您可以使用LinkedHashMap#removeEldestEntry虽然它没有提供根据条目在缓存内的时间量来删除的方法,但它确实为您提供了一种根据条目是否被访问来删除条目的方法。

它为实现者提供了在每次添加新条目时删除最旧条目的机会。如果映射表示缓存,这很有用:它允许映射通过删除过时的条目来减少内存消耗。

protected boolean removeEldestEntry(Map.Entry<K,V> eldest) {
  return size() > limit;

另请记住,您需要使用 初始化 LinkedHashMap accessOrder true

accessOrder - 排序模式 - 访问顺序为 true,插入顺序为 false。

所以把它们放在一起源应该如下所示

public class Cache<K, V> extends LinkedHashMap<K, V> {
    private final int MAX_ENTRIES = 100;
    public Cache() {
        super(16, 0.75f, true);// accessOrder is true
    }
    @Override
    protected boolean removeEldestEntry(Map.Entry<K, V> eldest) {
        return size() > MAX_ENTRIES;
    }
}
于 2012-10-19T10:02:54.860 回答
0

您可以只删除条件为真的缓存中的键。

于 2012-10-19T09:48:09.130 回答