先上代码:
import java.util.Map;
import java.util.WeakHashMap;
public class MyWeakHashMap {
public static void main(String[] args) {
Map<String, Integer> map = new WeakHashMap<>();
map.put(new String("1"), 1);
map.put(new String("2"),2 );
map.put(new String("3"),3 );
map.put("6",6);
String key = null;
for (String s : map.keySet()) {
// 这个"3"和new String("3")不是一个引用
if (s.equals("3")) {
key = s;
}
}
System.out.println("第一次打印"+map);
System.gc();
//取消清除掉1 2,3存活是因为3与key这个强引用挂钩,6存活是因为6存在于字符串常量池
map.put(new String("4"),4);
System.out.println("第二次打印"+map);
//key与3的引用断裂,再GC则3也会被回收
key = null;
System.gc();
System.out.println("第三次打印"+map);
}
}
结果如下:
运行结果图.png
解释如下:
第一次GC,即第二次打印的结果为上图是因为 1 2 是弱引用 ;而3没有消失是因为 key这个强引用与3挂钩,所以没有被GC掉;
第二次GC,即第三次打印的结果,是因为key=null,即key与3的引用断裂,所以3被GC掉,4被GC的原因与第一次一致,在这里就不解释了;
答主有点懵逼的地方就是:1 2 4明明是new String出来的,为什么会被GC?
后来答主明白了,在解释这个问题之前,我们先来补充一些知识:
JDK1.2后对引用进行了扩充,四种引用如下:
(1)强引用
使用最普遍的引用。如果一个对象具有强引用,它绝对不会被gc回收。如果内存空间不足了,gc宁愿抛出OutOfMemoryError,也不是会回收具有强引用的对象。
(2)软引用(SoftReference)
如果一个对象只具有软引用,则内存空间足够时不会回收它,但内存空间不够时就会回收这部分对象。只要这个具有软引用对象没有被回收,程序就可以正常使用。
(3)弱引用(WeakReference)
如果一个对象只具有弱引用,则不管内存空间够不够,当gc扫描到它时就会回收它。
(4)虚引用(PhantomReference)
如果一个对象只具有虚引用,那么它就和没有任何引用一样,任何时候都可能被gc回收。
然后我们来看WeakHashMap的小部分源码:
/**
* The entries in this hash table extend WeakReference, using its main ref
* field as the key.
*/
private static class Entry<K,V> extends WeakReference<Object> implements Map.Entry<K,V> {
V value;
final int hash;
Entry<K,V> next;
/**
* Creates new entry.
*/
Entry(Object key, V value,
ReferenceQueue<Object> queue,
int hash, Entry<K,V> next) {
super(key, queue);
this.value = value;
this.hash = hash;
this.next = next;
}
从Entry的构造方法我们知道,key最终会传到到Reference的构造方法中,但是我们更需要注意的是,Entry继承了WeakReference这个类,也就是上文提到的弱引用,再结合上文”如果一个对象只具有弱引用,则不管内存空间够不够,当gc扫描到它时就会回收它。”
似乎还是没解决“ 1 2 4明明是new String出来的,为什么会被GC ”这个问题,但是我们思考这样一个问题,虽然这个对象是new出来的,但是如果这个对象同时也继承了WeakReference,即与弱引用挂钩,那这个对象到底是属于强引用还是弱引用呢?
答主先回答“ 1 2 4明明是new String出来的,为什么会被GC ”这个问题,
虽然是new出来的对象,但是new的引用 在传递参数之后就丢失了,而与它继承的WeakReference挂上了钩,所以属于弱引用对象,GC的时候,会被回收。
网友评论