一尘不染

如何清理ThreadLocals

tomcat

有没有人举一个例子呢?它们由垃圾收集器处理吗?我正在使用Tomcat 6。


阅读 1577

收藏
2020-06-16

共1个答案

一尘不染

Javadoc这样说:

“只要线程处于活动状态并且ThreadLocal实例是可访问的,每个线程都会对其线程本地变量的副本进行隐式引用;线程消失后,其线程本地实例的所有副本都将进行垃圾回收。
(除非存在对这些副本的其他引用)。

如果您的应用程序或(如果您在谈论请求线程)容器使用线程池,则意味着线程不会死亡。如有必要,您将需要自己处理线程本地对象。唯一干净的方法是调用该ThreadLocal.remove()方法。

您可能要清理线程池中的线程的线程本地有两个原因:

  • 防止内存(或假设的资源)泄漏,或
  • 以防止通过线程本地将信息从一个请求意外泄漏到另一个请求。

对于有界线程池,线程本地内存泄漏通常不应成为主要问题,因为任何线程本地变量最终都可能被覆盖。即当线程被重用时。但是,如果您犯ThreadLocal了一次又一次地创建新实例的错误(而不是使用static变量来保存单例实例),则线程局部值将不会被覆盖,并且会在每个线程的threadlocals映射中累积。这可能会导致严重的泄漏。


假设您要谈论的是在Web应用程序处理HTTP请求期间创建/使用的线程本地,那么避免线程本地泄漏的一种方法是在ServletRequestListener您的Web应用程序中注册一个,ServletContext并实现侦听器的requestDestroyed方法来清理该线程本地。当前线程。

请注意,在这种情况下,您还需要考虑 信息 从一个请求泄漏到另一个请求的可能性。

2020-06-16