整体架构
强引用
- 当内存不足的时候,JVM 便开始对垃圾进行回收,对应强引用的对象,即使出现了OOM(内存溢出) , 也不会回收。
强引用是我们最常见的普通对象引用,只要还有一个强引用指向一个对象,就说明这个对象还活着,垃圾回收器不会回收此对象。
在Java 中我们直接new一个对象,当把这个对象赋给一个引用变量,则这个引用变量就是一个强引用。当一个对象处于强引用的时候,表名到达GCRoot是可达的状态,它是不可能被垃圾回收期回收的。即使这个对应以后不会使用到,只要有这个强引用在,就不会被回收,因此,强引用是造成java 内存泄露的主要原因之一。
Object obj1 = new Object();//这样定义的默认就是强引用
Object obj2 = obj1;//obj2就是引用赋值
obj1 = null;//设置为null
System.gc();
System.out.println(obj2);
对于一个普通的Java对象,如果没有其他的引用关系,即使用完了就可以设置为null,
一般认为就是可以被垃圾收集,当然时候时候回收还得要看垃圾回收器。
软引用
软引用需要用 java.lang.ref.softReference 类来实现
对于软引用的对象要记住以下两点:
- 当系统内存充足时,不会被回收。
- 当系统内存不足时,则会被回收
软引用通常使用在内存比较敏感的程序中,比如高速缓存就有用到软引用个, 内存够用的时候就保留,不够用的时候就回收。
Object obj1 = new Object();
SoftReference<Object> softReference = new SoftReference<>(obj1);
System.out.println(obj1);
System.out.println(softReference.get());
obj1=null;
System.gc();
System.out.println(obj1);
System.out.println(softReference.get());
弱引用
Object o1 = new Object();
WeakReference weakReference = new WeakReference<>(o1);
System.out.println(o1);
System.out.println(weakReference.get());
o1=null;
System.gc();
System.out.println("================");
System.out.println(o1);
System.out.println(weakReference.get());
结果如下
java.lang.Object@7ea987ac
java.lang.Object@7ea987ac
================
null
null
也就是说,对于软引用,一旦发生GC,并且 WeakReference 包裹的元素没有强引用,则就会被GC回收。
其实GC回收后,把 Weakreference 放到了 引用队列中保存了下来
Object o1 = new Object();
ReferenceQueue<Object> referenceQueue = new ReferenceQueue<>();
WeakReference weakReference = new WeakReference<>(o1,referenceQueue);
System.out.println(o1);
System.out.println(weakReference.get());
System.out.println(referenceQueue.poll());
o1=null;
System.gc();
System.out.println("================");
System.out.println(o1);
System.out.println(weakReference.get());
System.out.println(referenceQueue.poll());
测试结果:
java.lang.Object@7ea987ac
java.lang.Object@7ea987ac
null
================
null
null
java.lang.ref.WeakReference@12a3a380
弱引用使用场景: