如何配置Spring 4.0.5以使用@ExceptionHandler而不是ExceptionResolver?


问题内容

我在特定控制器上添加了@ExceptionHandler方法。

@ExceptionHandler(NullPointerException.class)
@ResponseStatus(HttpStatus.Not_FOUND)
public @ResponseBody doSomeThing(NullPointerException e) {...}

问题是:我也有一个习惯

org.springframework.web.servlet.handler.AbstractHandlerExceptionResolver

在我的Rest Application(旧版)中定义为@Component。

现在,我的特定控制器中的@ExceptionHandler永远不会被调用。我需要支持两种异常处理策略。

如何配置spring 4.0.5,以便调用我的Controller中的@ExceptionHandler?


问题答案:

DispatcherServlet默认情况下将加载并注册所有HandlerExceptionResolver范围内的实现ApplicationContext,以帮助您处理例外Controllers。您会注意到AbstractHandlerExceptionResolver该类实现了Ordered。大多数HandlerExceptionResolver实现会扩展此类,并DispatcherServlet使用HandlerExceptionResolver它来对处理您的抛出的异常进行排序Controllers

您可能会怀疑,@ExceptionHandler您的批注Controllers处理也由HandlerExceptionResolver实现来处理。确切地说:org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerExceptionResolver

我建议在您的情况下发生的是,HandlerExceptionResolver通过DispatcherServlet引起的排序会导致您的实现在有效忽略注释的AnnotationMethodHandlerExceptionResolver含义之前被调用@ExceptionHandler

为了解决这个问题,我们需要做两件事:

  1. 直接AnnotationMethodHandlerExceptionResolver在您的Bean中注册为BeanApplicationContext
  2. Order将此bean 的值设置为大于自定义HandlerExceptionResolver实现的值

这意味着在处理Exceptions您的时Controllers,Spring将首先查看AnnotationMethodHandlerExceptionResolvercan
是否可以处理Exception,如果不能,则退回到您的自定义HandlerExceptionResolver实现。

作为设置顺序的示例HandlerExceptionResolver(假设您在某处使用XML配置):

<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerExceptionResolver" p:order="1" />

您可能还需要查看默认情况下HandlerExceptionResolver注册的实现列表,DispatcherServlet并确定最适合您的应用程序的顺序。