Thread Local内存泄漏问题


登录功能之Thread Local

功能

为了解决用户登录后保存登录信息的功能

  1. 为什么不用Session?

  2. Thread Local比Session好在哪

原理

每个Thread内部都有一个Map,我们每当定义一个ThreadLocal变量,就相当于往这个Map里放了一个key,并定义一个对应的value。每当使用ThreadLocal,就相当于get(key),寻找其对应的value

如图所示:

img

实线代表强引用,虚线代表弱引用

  • 每一个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不会被回收,久而久之就会发生内存泄漏问题