Thread Local内存泄漏问题
功能
为了解决用户登录后保存登录信息的功能
-
为什么不用Session?
-
Thread Local比Session好在哪
原理
每个Thread内部都有一个Map,我们每当定义一个ThreadLocal变量,就相当于往这个Map里放了一个key,并定义一个对应的value。每当使用ThreadLocal,就相当于get(key),寻找其对应的value
如图所示:
实线代表强引用,虚线代表弱引用
-
每一个Thread维护一个ThreadLocalMap, key为使用弱引用的ThreadLocal实例,value为线程变量的副本。
-
强引用,使用最普遍的引用,一个对象具有强引用,不会被垃圾回收器回收。当内存空间不足,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不回收这种对象。
-
如果想取消强引用和某个对象之间的关联,可以显式地将引用赋值为null,这样可以使JVM在合适的时间就会回收该对象。
-
弱引用,JVM进行垃圾回收时,无论内存是否充足,都会回收被弱引用关联的对象。在Java中,用java.lang.ref.WeakReference类来表示。
内存泄漏问题
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
//如果不删除 ThreadLocal中用完的信息 会有内存泄漏的风险
UserThreadLocal.remove();
}
为什么用完之后要回收线程呢?
如图所示,因为回收机回收的时候弱引用会被回收,但是强引用不会,所以当回收机回收时候Key被回收了,但是线程不会被回收,也就是说value不会被回收,久而久之就会发生内存泄漏问题