1. 强引用 当内存不足,JVM开始垃圾回收,对于强引用的对象,就算是出现OOM也不会对该对象进行回收。
强引用是最常见的普通对象引用,只要还有强引用指向一个对象,就表明这个对象还活着,垃圾回收器就不会回收这种对象。在Java中,把一个对象赋给一个引用变量,这个引用变量就是一个强引用。当一个对象被强引用时,它处于可达状态,它是不可能被垃圾回收机制回收的。因此强引用是造成Java内存泄漏的主要原因之一。
对于一个普通对象,如果没有其它引用关系,只有超过了引用的作用域或者显式地将强引用赋值为null,一般热认为可以被垃圾回收了。
2. 软引用 软引用是一种相对强引用弱化了一些的引用,需要java.lang.ref.SoftReference类来实现,可以让对象豁免一些垃圾回收器。对于只有软引用的对象,当系统内存充足时,它不会被回收,当系统内存不足的时候,会被回收。
软引用通常用在对内存敏感的程序中,比如高速缓存就有用到软引用,内存够用时保留,不够时回收。(mybatis缓存)
Object o1=new Object();
SoftReference<Object> softReference = new SoftReference<>(o1);
System.out.println(softReference.get());
3. 弱引用 弱引用需要java.lang.ref.WeakReference类来实现,它比软引用的生存期更短,对于弱引用的对象,只要垃圾回收机制运行,不管JVM的内存空间是否足够,都会回收该对象占用的内存。(ThreadLocalMap里面用了)
WeakHashMap,垃圾回收机制运行,会回收占用的内存
4. 虚引用 虚引用需要java.lang.ref.PhantomReference类来实现 虚引用与其他引用不同,虚引用并不会决定对象的生命周期。
对象仅持有虚引用,在任何时候都可能被垃圾回收器回收,它不能单独使用,也不能通过它访问对象,虚引用必须和引用队列(ReferenceQueue)联合使用。
虚引用的主要作用是跟踪对象被垃圾回收的状态,仅仅是提供了一种确保对象被finalize以后,做某些事情的机制。
PhantomReference.get方法总是返回null,因此无法访问对应的引用对象。其意义在于说明一个对象已经进入finalization阶段,可以被gc回收,用于实现比finalization机制更灵活的回收操作。
设置虚引用的唯一目的,就是在这个对象被收集器回收时收到一个系统通知或后续添加进一步的处理。Java技术允许使用finalize方法在垃圾收集器将对象从内存中清楚出去之前做必要的清理工作。
Object o1=new Object();
ReferenceQueue<Object> referenceQueue = new ReferenceQueue<>();
PhantomReference<Object> phantomReference = new PhantomReference<>(o1,referenceQueue);
System.out.println(phantomReference.get());
Java中可以作为GC Roots的对象
- 虚拟机栈中的引用对象
- 方法区中的类静态属性引用的对象(一般指被static修饰的对象,加载类的时候就加载到内存中)
- 方法区中的常量引用的对象
- 本地方法栈中的JNI(native方法)引用的对象
|