7

WeakHashMap 中的关键对象变得弱可达。并且 map 应该在 GC 之后删除条目。但是对值对象的强引用仍然存在。为什么?

使用番石榴弱键映射观察到相同的行为。

预期输出:

...
refKey.get = null
refValue.get = null

但我得到输出:

map.keys = []
map.values = []
map.size = 0
refKey.get = null
refValue.get = (123)

代码:

import java.lang.ref.WeakReference;
import java.util.Map;
import java.util.WeakHashMap;
import com.google.common.collect.MapMaker;

public class Test {

    static class Number {
        final int number;
        public Number(int number) { this.number = number; }
        public String toString() { return "(" + number + ")"; }
    }

    static class Key extends Number {
        public Key(int number) { super(number); }
    }

    static class Value extends Number {
        public Value(int number) { super(number); }
    }

    public static void main(String args[]) {

        //Map<Key, Value> map = new MapMaker().weakKeys().makeMap();
        Map<Key, Value> map = new WeakHashMap<>();

        Key key = new Key(1);
        Value value = new Value(123);

        map.put(key, value);

        WeakReference<Key> refKey = new WeakReference<>(key);
        WeakReference<Value> refValue = new WeakReference<>(value);

        key = null;
        value = null;

        System.gc();

        System.out.println("map.keys = " + map.keySet());
        System.out.println("map.values = " + map.values());
        System.out.println("map.size = " + map.size());
        System.out.println("refKey.get = " + refKey.get());
        System.out.println("refValue.get = " + refValue.get());

    }

}

升级版:

我尝试在 jСonsole 和 jcmd 中执行 GC,但输出没有改变。

4

3 回答 3

7

WeakHashMap包含Map.Entry使用 a 引用密钥的实例(WeakReference实际上,在 OpenJDK / Oracle JDK 中,它直接 extendsWeakReference)。

当 GC 发生时,现在引用缺失键的条目不会从映射中神奇地删除:它们仍然存在直到被清除,这就是为什么值仍然存在并且尚未被收集的原因。

在 OpenJDK 中,这发生在expungeStaleEntries()使用 aReferenceQueue时,并且从多个地方调用该方法:

  • size()
  • resize()
  • getTable()它本身是从多种方法调用的,包括get()put()

如果您希望您的值可被垃圾回收,您应该与 进行交互WeakHashMap,例如通过询问size()或进行查找。

请注意,这意味着在第二次垃圾收集之前无法收集该值。

如果我没记错的话,它在 Guava 中的工作方式或多或少是相同的。

于 2016-01-19T14:28:23.473 回答
2

运行此代码修改并在 Jconsole 中强制 GC 确认删除了引用。

    System.gc();

    try {
        while (refValue.get() != null) {
            System.out.println("map.keys = " + map.keySet());
            Thread.sleep(5000);
        }
    } catch (InterruptedException e) {
        e.printStackTrace();
    }

    System.out.println("map.keys = " + map.keySet());
    System.out.println("map.values = " + map.values());
    System.out.println("map.size = " + map.size());
    System.out.println("refKey.get = " + refKey.get());
    System.out.println("refValue.get = " + refValue.get());

但是由于某种原因,如果map.keySet()没有在循环中查询,它永远不会终止,即refValue不会成为null.

于 2016-01-19T14:18:53.623 回答
1

您无法控制 Java 中的 GC。虚拟机决定。我从来没有遇到过需要 System.gc() 的情况。由于 System.gc() 调用只是简单地建议 VM 进行垃圾收集并且它还进行完整的垃圾收集(多代堆中的旧代和新代),因此它实际上可能导致消耗更多的 cpu 周期必要的。

在 jdk 1.7 以后,您可以使用以下命令强制 gc 运行

jcmd <pid> GC.run

获取 processid 使用jps

于 2016-01-19T13:38:32.050 回答