我有以下类:
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 更好,因为即使处理程序无法正确处理请求(也称为发生异常),也会调用它。