删除Spring MVC网站中的ThreadLocal对象



我有以下类:

public class AppContext {
        
    private static final ThreadLocal<AppContextData> contextHolder = new ThreadLocal<AppContextData>() {
        
        @Override
        protected AppContextData initialValue() {
            return new AppContextData();
        }
    };
    
    public static AppContextData get() {
        return contextHolder.get();
    }
    
    public static void unset() {
        contextHolder.remove();
    }
}
public class AppContextData {
   //many getter and setter methods
}

网站开始在拦截器中使用 ThreadLocal 对象,并且该对象在整个 Web 请求中使用。

public class MyIntercepter extends HandlerInterceptorAdapter {
            
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)  {
    
        AppContext.get();
        //..... 
    }
}

我在Tomcat 8.x上运行这个网站,我发现日志有以下错误消息。

20-May-2016 22:43:53.657 严重 [localhost-startStop-2] org.apache.catalina.loader.WebappClassLoaderBase.checkThreadLocalMapForLeaks Web 应用程序 [ROOT] 创建了一个 ThreadLocal 键类型为 [my.organization.data.AppContext$1](值 [my.organization.data.AppContext$1@31d4463a])和类型为 [my.organization.data.AppContextData] 的值(值 [my.organization.data.AppContextData@e89d5f4]),但在 Web 应用程序停止时未能将其删除。线程将随着时间的推移而更新,以避免可能的内存泄漏。

如何解决此问题?

JanPi 已经在评论中说了我们必须做什么。如果有人想知道如何,这里是:

public class MyIntercepter extends HandlerInterceptorAdapter {
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)  {
        AppContext.get();
        //..... 
    }
    public void afterCompletion(
        HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        AppContext.unset();
    }
    public void afterConcurrentHandlingStarted(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception{
        // only relevant when a handler starts an asynchronous request.
        AppContext.unset();
    }
}

postHandle 方法看起来是个好地方,但 afterCompletion 更好,因为即使处理程序无法正确处理请求(也称为发生异常),也会调用它。

最新更新