Java中几个Reference作用,也是面试的时候经常问到的问题,以前总是记一次忘一次,现在有时间,索性写个demo测试一把。
具体代码如下:
JVM 参数:-Xmx10m -Xms5m -XX:+PrintGC
SoftReference的时候:
weakReference的时候:
StrongReference:
由于strong是JVM默认的,这里就不做了,直接就是一点都不会被回收,直至OOM
PhantomReference:
虚引用并不会改变内存回收机制,只是在回收的时候放到ReferenceQueue里通知用户可做一些额外操作,比如打印日志等
如下代码,本质上byte[]还是一个强引用的。
输出:
总结一下:
StrongReference:JVM默认,除非GC时已无任何对象引用,否则即便是OOM也不会回收
WeakReference:一旦没有被引用,GC时就会回收
SoftReference:和strong一样,除非GC时已无任何对象引用,否则....在即将OOM前才会被回收,所以SoftReference一般可以用来做缓存
sun.nio.ch.Util中就有private static ThreadLocal<SoftReference<SelectorWrapper>> localSelector
= new ThreadLocal<SoftReference<SelectorWrapper>>();
用于临时的Selector。
PhantomReference:通过构造函数的ReferenceQueue作为一个通知,用来在对象被回收时做额外的操作。
更新:
对于weakReference的理解有偏差,上面的测试和strongReference没啥差别,重新加了测试:
上图中不将b的引用去除掉(b=null),增加循环
结果:
执行了2次不同的循环次数后jvm执行gc,这时将wr中的byte[]回收掉。
- WeakReference的一个特点是它何时被回收是不可确定的, 因为这是由GC运行的不确定性所确定的. 所以, 一般用weak reference引用的对象是有价值被cache, 而且很容易被重新被构建, 且很消耗内存的对象.
总结来说wr是一种比sr更弱的缓存,当某些对象最好能被缓存,但是重新构建问题也不大的情况下可以使用。