本文共 2232 字,大约阅读时间需要 7 分钟。
看一个例子:
class Stack { private Object[] elements; private int size = 0; private static final int DEFAULT_INITIAL_CAPACITY = 16; public Stack() { elements = new Object[DEFAULT_INITIAL_CAPACITY]; } public void push(Object e) { ensureCapacity(); elements[size++] = e; } public Object pop() { if (size == 0) throw new EmptyStackException(); return elements[--size]; } public void ensureCapacity() { if (elements.length == size) elements = Arrays.copyOf(elements, 2 * size + 1); } }
这段代码存在一个内存泄漏问题:如果一个栈先增长,再收缩,那么从栈中弹出的元素不会被当做垃圾回收;即使栈的程序不在引用这些对象,他们也不会被回收.
这是因为,栈内部维护着这些对象的过期引用;过期引用是指永远不再会被解除的引用. 内存泄漏的来源可能:修复:一旦对象引用已经过期,只需清理这些引用即可;对于上面的例子而言,一旦元素从栈中弹出,指向它的引用就过期了;pop方法修改如下:
public Object pop() { if (size == 0) throw new EmptyStackException(); Object result = elements[--size]; element[size]=null//消除过期引用 return result ; }
import java.util.HashMap;import java.util.Map;import java.util.Map.Entry;import java.util.WeakHashMap;public class WeakHashMapTester { static MapwMap = new WeakHashMap (); static Map map = new HashMap (); static { String ref1 = new String("obejct1"); String ref2 = new String("obejct2"); wMap.put(ref1, "chaheObject1"); map.put(ref2, "chaheObject2"); } public static void TestWeahHashMap() { System.out.println("WeakHashMap GC之前"); for (Entry str : wMap.entrySet()) { System.out.println(str); } System.gc(); System.out.println("WeakHashMap GC之后"); for (Entry str : wMap.entrySet()) { System.out.println(str); } } public static void TestHashMap() { System.out.println("HashMap GC之前"); for (Entry str : map.entrySet()) { System.out.println(str); } System.gc(); System.out.println("HashMap GC之后"); for (Entry str : map.entrySet()) { System.out.println(str); } } public static void main(String[] args) { TestWeahHashMap(); System.out.println("---------"); TestHashMap(); }}
结果:
WeakHashMap GC之前obejct1=chaheObject1WeakHashMap GC之后---------HashMap GC之前obejct2=chaheObject2HashMap GC之后obejct2=chaheObject2
更多:
Effective java转载地址:http://glwji.baihongyu.com/