thunder_yan 发表于 2013-2-3 10:40:33

传奇的WeakHashMap

在广大的Java界,关于WeakHashMap一直都存在这么个传说:

可是WeakHashMap是真的自动移除其条目吗?
 
今天因为闲来无事,所以想看看WeakHashMap是如何自动实现移除其内部不用的条目从而达到的自动释放内存的目的的。仔细的看了看JVM自带的源代码的实现,在WeakHashMap是主要通过expungeStaleEntries这个函数的来实现的。基本上只要对WeakHashMap的内容进行访问就会调用这个函数,从而达到清除其内部不在为外部引用的条目。但是如果预先生成了WeakHashMap,而在GC以前又不曾访问该WeakHashMap,那不是就不能释放内存了吗?
 
写个代码测试一把:

public static void main(String[] args) throws Exception {      List<WeakHashMap<byte[][], byte[][]>> maps = new ArrayList<WeakHashMap<byte[][], byte[][]>>();      for (int i = 0; i < 1000; i++) {          WeakHashMap<byte[][], byte[][]> d = new WeakHashMap<byte[][], byte[][]>();          d.put(new byte, new byte);          maps.add(d);          System.gc();          System.err.println(i);          }    } 

由于Java默认内存是64M,所以再不改变内存参数的情况下,该测试跑不了几步循环就内存溢出了。果不其然,WeakHashMap这个时候并没有自动帮我们释放不用的内存。
 
再加个对会对map进行访问的测试试试:

public static void main(String[] args) throws Exception {            List<WeakHashMap<byte[][], byte[][]>> maps = new ArrayList<WeakHashMap<byte[][], byte[][]>>();            for (int i = 0; i < 1000; i++) {            WeakHashMap<byte[][], byte[][]> d = new WeakHashMap<byte[][], byte[][]>();            d.put(new byte, new byte);            maps.add(d);            System.gc();            System.err.println(i);                for (int j = 0; j < i; j++) {                  System.err.println(j+" size" + maps.get(j).size());            }          }      } 这下测试就顺利通过了。
页: [1]
查看完整版本: 传奇的WeakHashMap