4

我想测试 Java WeakHashMap 类的功能,为此我编写了以下测试:

public class WeakHashMapTest {

public static void main(String args[]) {
    Map<String, Object> weakMap = new WeakHashMap<>();
    String x = new String("x");     
    String x1 = new String("x1");   
    String x2 = new String("x2");   
    weakMap.put(x, x);
    weakMap.put(x1, x1);
    weakMap.put(x2, x2);
    System.out.println("Map size :" + weakMap.size());
    // force all keys be eligible
    x=x1=x2=null;
    // call garbage collector 
    System.gc();
    try {
        Thread.sleep(1000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }

    System.out.println("Map size :" + weakMap.size());
    System.out.println("Map :" + weakMap);
}

}    

运行类 WeakMapTest 后,我​​很惊讶地得到以下输出:

gc 前的映射:{x=x, x1=x1, x2=x2} gc 后的映射:{x=x, x1=x1, x2=x2}

当我预计地图将为空时。

也就是说,垃圾收集器没有做它的工作。但为什么?

4

2 回答 2

5

WeakHashMap它们不再强可达时,垃圾收集器将回收其密钥。

实现说明: WeakHashMap 中的值对象由普通的强引用保存。因此,应注意确保值对象不会直接或间接地强烈引用它们自己的键,因为这将防止键被丢弃。

但是,由于您将键本身用作值,因此该值仍然是强可达的,这意味着垃圾收集器无法回收键。

但是,如果您使用其他对象作为值,则对键的唯一引用将是键本身。

weakMap.put(x, new Object());
weakMap.put(x1, new Object());
weakMap.put(x2, new Object());

然后,在清除变量并像您已经完成的那样调用垃圾收集器之后,我得到了输出:

Map size :3
Map size :0
Map :{}

即使调用System.gc()不能保证垃圾收集器运行,看起来它确实在这里运行。

于 2015-01-06T01:37:56.780 回答
1

System.gc()实际上是运行垃圾收集器的建议。没有保证强制垃圾收集器运行的方法。

于 2015-01-06T01:34:09.310 回答