问题
自己写了一个通用plugin工具包,其中一项是集成了jsr303进行参数验证,只有引入myplugin-validate-starter就能瞬间集成参数验证功能无需再写代码,然而开始集成时一切OK,但是当我在项目中写了项目本身的全局异常拦截后就出问题,参数验证异常未生效。
原因
上网查询原因后发现是异常优先级的问题,根据查询的信息给关键代码打上断点,
# 关键方法
ExceptionHandlerExceptionResolver.doResolveHandlerMethodException()
// 关键代码
for (Map.Entry<ControllerAdviceBean, ExceptionHandlerMethodResolver> entry : this.exceptionHandlerAdviceCache.entrySet()) {
ControllerAdviceBean advice = entry.getKey();
if (advice.isApplicableToBeanType(handlerType)) {
ExceptionHandlerMethodResolver resolver = entry.getValue();
Method method = resolver.resolveMethod(exception);
if (method != null) {
return new ServletInvocableHandlerMethod(advice.resolveBean(), method, this.applicationContext);
}
}
}
查看this.exceptionHandlerAdviceCache发现存在2个ExceptionHandler,发现我项目中定义的全局异常拦截拦截了Exception异常,而且刚好项目ExceptionHandler排在validateExceptionHandler前面,谜题总算是揭开了。
解决
现在的问题就是如何让参数异常在exception前面被处理。
可以使用@order注解来给handler排序,但是问题是我希望给异常拦截排序,但是这个排序是给handler排序的,而且项目本身的异常拦截器优先级应该更高,这样就无法让exception拦截排到参数验证异常拦截后面。除非将exception异常拦截与其它业务异常分开用单独的一个类写,然后给与其最低的优先级。