强引用
宁愿OOM,也不回收。
软引用
内存够,不回收,内存不够,回收。
Object o = new Object();
SoftReference<Object> reference = new SoftReference<>(o);
System.out.println(reference.get());
try {
o = null;
byte[] bytes = new byte[1024 * 1024 * 10];
System.gc();//当内存不够时,可注释,系统会自动调用垃圾回收算法
}
catch (Exception e) {
}
finally {
System.out.println(reference.get());
}
当内存足够时
返回结果:
java.lang.Object@6842775d
java.lang.Object@6842775d
当分配内存不哆时,配置jvm参数 -Xmx10m
返回结果:
java.lang.Object@17ed40e0
null
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at test.TestMain.main(TestMain.java:20)
弱引用
无论内存是否够用,都回收。
Object o = new Object();
WeakReference<Object> weakReference = new WeakReference<>(o);
System.out.println(weakReference.get());
o = null;
System.gc();
System.out.println(weakReference.get());
返回结果:
java.lang.Object@6842775d
null
应用场景:
假如一个应用需要读取大量的本地图片:
1. 如果每次读取图片都从硬盘读取则会严重影响性能
2. 如果一次性全部加载到内存又可能造成内存溢出
3. 设计思路:用一个hashMap来保存图片的路径和相应图片对象关联的软引用之间的映射关系,在内存不足时,JVM会自动回收这些缓存图片对象所占用的空间,从而有效的避免了OOM的问题。
4. HashMap<String, SoftReference<Bitmap>> imageCache = new HashMap<>();
WeakHashMap
WeakHashMap<Integer, String> map = new WeakHashMap<>();
Integer key = new Integer(1);
String value = "abc";
map.put(key, value);
System.out.println(map);
key = null;
System.gc();
System.out.println(map);
返回结果
{1=abc}
{}
虚引用
- 它不能单独使用,也不能通过它访问对象,虚引用必须与引用队列(ReferenceQueue)联合使用
- 虚引用的主要作用是跟踪对象被垃圾回收的状态。仅仅提供了一种确保对象被finalize以后,做某些事情的机制。
- 换句话说,设置虚引用关联的唯一目的,就是在这个对象被收集器回收的时候收到一个系统通知或者后续添加进一步处理。Java技术允许使用finalize()方法在垃圾收集器将对象从内存中清除出去之前做必要的清理工作。
Object o = new Object();
ReferenceQueue<Object> referenceQueue = new ReferenceQueue<>();
PhantomReference<Object> phantomReference = new PhantomReference<>(o, referenceQueue);
System.out.println(phantomReference.get());
System.out.println(referenceQueue.poll());
System.out.println("...............");
o = null;
System.gc();
System.out.println(o);
System.out.println(phantomReference.get());
System.out.println(referenceQueue.poll());
返回结果
null
null
...............
null
null
java.lang.ref.PhantomReference@6842775d