一个由单例模式在多线程环境下引发的 bug

问题症状

HTTP 日志系统,老是出现日志信息覆盖的情况。比如同时调用 A 接口和 B 接口,B 接口请求响应信息变成了 A 接口请求响应相关信息。这个问题在并发量大的情况下越来越严重。

问题初步分析

显然并发量越来越大,问题越来越严重,是一个多线程问题。日志采集是通过 Spring 的 LogHttpInterceptor 来做的,分析一下代码。

public class LogHttpInterceptor extends HandlerInterceptorAdapter {

    private Logger logger;
    private PathMatcher pathMatcher;
    private String[] excludePaths;
    private LogMsg msg;

    public LogHttpInterceptor(Logger logger) {
        this.logger = logger;
    }

    public LogHttpInterceptor(Logger logger, String... excludePaths) {
        this.logger = logger;
        this.excludePaths = excludePaths;
        if (!StringUtils.isEmpty(this.excludePaths)) {
            pathMatcher = new AntPathMatcher();
        }
    }

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response,
                             Object handler) throws Exception {

        if (isSkip(request)) {
            return true;
        }
        msg = new LogMsg(logger.getModule());
        msg.putRequest(request);
        LogMsgThreadMapper.putLogMsg(LogMsg.REQUEST_TIME, "" + System.currentTimeMillis());
        return true;
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response,
                                Object handler, Exception ex) throws Exception {
        if (isSkip(request)) {
            return;
        }
        LogMsgThreadMapper.putLogMsg(LogMsg.RESPONSE_TIME, "" + System.currentTimeMillis());
        msg.putResponse(request, response);
        if (ex != null) {
            msg.append(ex.getMessage());
        }
        logger.log(msg);
    }

    private boolean isSkip(HttpServletRequest request) {
        if (pathMatcher != null && excludePaths != null) {
            for (String exclude : excludePaths) {
                if (pathMatcher.match(exclude, request.getRequestURI())) {
                    return true;
                }
            }
        }
        return false;
    }
}

其实我看了域变量 private LogMsg msg 我就感觉有问题了。在一个 Spring 框架中,像 HandlerInterceptorAdapter 一般都会生成单例 Bean。我仔细研读了注册 HandlerInterceptorAdapter Bean 代码,果然。

   @Bean
    public HandlerInterceptorAdapter aliLoggerInterceptor(@Autowired AliLogger aliLogger) {
        LogHttpInterceptor interceptor = new LogHttpInterceptor(aliLogger, "/api/s/s");
        return interceptor;
    }

单例 Bean 在多线程环境下,写域变量是一件很危险的事情——每个线程都可以修改域变量的值。仔细看一下 LogMsg 使用,首先在 preHandle new 出一个实例,再在 afterCompletion 继续使用,在传入日志系统进行日志处理。

什么情况会出现该问题呢?具体分析一下。

  • 第一种情况,请求 A preHandle、afterCompletion处理完,请求 B 继续处理 preHandle、afterCompletion,LogMsg 是新的,不是这种情况
  • 第二种情况,请求 A preHandle 处理完、afterCompletion未处理完,请求 B 处理 preHandle,无论请求 B afterCompletion 是否处理完,请求 A 的 LogMsg 被修改

复现

复现该问题也简单,就不贴代码了。

  • 在 Controller 中新建两个接口,一个叫 /fast,一个叫 /slow
  • /slow 先休眠 10 s,/fast 立刻返回
  • 先调用 /slow 接口,再调用 /fast 接口

问题必现了。

修复

很简单,不要在单例中共享对象。实现对象传递也要在 ThreadLocal 中。此问题只要把 LogMsg 放在 ThreadLocal 中操作即可,线程执行结束或者开始时,清理一下 ThreadLocal。

总结

  • 单例模式不要在域中共享变量
  • 线程共享变量最好在 ThreadLocal 中,以并发集合传递数据也是种不错的选择
  • 对于多线程,要小心谨慎
posted @ 2018-07-19 01:01  Piers  阅读(859)  评论(0编辑  收藏  举报