图片来自尚硅谷视频
1.强引用,弱引用,软引用,虚引用
强引用(Strong Reference)是Java中最为普遍的引用类型。当一个对象被强引用关联时,垃圾回收器不会回收这个对象,即使系统内存不足也不会回收。只有当该对象的强引用被显式地释放,或者不再被任何引用关联时,该对象才会成为垃圾回收的候选对象。
软引用用于描述一些还有用但并非必须的对象,在内存不足时可能被垃圾回收。
弱引用(Weak Reference)是Java中一种比强引用更弱的引用类型。当一个对象只被弱引用关联时,在下一次垃圾回收时,该对象就有可能被回收。垃圾回收器会在适当的时候回收仅被弱引用持有的对象,即使内存并不紧张。
虚引用(Phantom Reference)是Java中最弱的引用类型之一,无法通过引用直接获取到对象实例。虚引用主要用于跟踪对象被垃圾回收的状态。当一个对象只被虚引用关联时,其实际上并不影响对象的生命周期,也就是说,垃圾回收器随时可能回收被虚引用关联的对象。
2.ThreadLocal内存泄漏问题
弱引用使用方法: WeakReference weakRef = new WeakReference<>(obj);
Entry继承WeakReference调用WeakReference的构造方法super(v),
等价于 WeakReference weakRef = new WeakReference<>(ThreadLocal);
3.引用接口关系图
Reference:强引用
SoftReference:软引用
WeakReference:弱引用
PhantomReference:虚引用
3.引用实战
1.强引用
2.软引用:结果不会打印,不会回收软引用(弱引用类似)
3.虚引用使用注意事项,以及实战
注意事项:虚引用被回收会走进引用队列里
api
4.软引用弱引用 引用场景
5.源码为什么使用弱引用吗
假设ThreadLocal是局部变量,强引用在栈上面会被销毁,但如果Entry的Key在堆上还是强引用 指向ThreadLocal,ThreadLocal就无法被回收
使用弱引用还会造成内存泄漏,为什么?(如上图所示)
Key为null的时候,大部分场景都会用线程池,就导致线程复用,有一条指向map到entry可达性链条无法被GC,但value无法被访问导致内存泄漏
value为什么不适用弱引用
GC的时候value就被变成null,导致可能突然系统报错