如何让一个由 ControllerAdvice 注释的类来处理不同类型的异常?

Posted

技术标签:

【中文标题】如何让一个由 ControllerAdvice 注释的类来处理不同类型的异常?【英文标题】:How to have a class annotated by ControllerAdvice to handle different types of exceptions? 【发布时间】:2015-09-20 17:42:14 【问题描述】:

我有一个名为GlobalExceptionHandler 的类,由ControllerAdvice 注释。它正确处理所有NoHandlerFoundExceptions。我添加了一个新方法来处理InternalError 异常,但它不处理此类异常;因此,我仍然收到HTTP Status 500

基于这个link,500 (Internal Server Error) 的异常类是ConversionNotSupportedException

我尝试了以下代码,但没有捕获内部服务器错误。

1

@ControllerAdvice
public class GlobalExceptionHandler 
    @ExceptionHandler
    @ResponseStatus(HttpStatus.NOT_FOUND)
    public String handleExceptionNotFound(NoHandlerFoundException ex) 
        System.err.println("not found");
        return "redirect:/error";
    

    @ExceptionHandler(ConversionNotSupportedException.class)
//  @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
    public String handleExceptionsInternalError(ConversionNotSupportedException ex) 
        ex.printStackTrace();
        System.err.println("internal error");
        return "redirect:/error";
    


2

@ExceptionHandler(ConversionNotSupportedException.class)
public String handleExceptionsInternalError(HttpServletRequest req, ConversionNotSupportedException ex) 

3

@ExceptionHandler
@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
public String handleExceptionsInternalError(ConversionNotSupportedException ex) 

我需要处理的异常如下:

HTTP Status 500 - Could not resolve view with name 'user' in servlet with name 'myproject'
javax.servlet.ServletException: Could not resolve view with name 'user' in servlet with name 'myproject'
    org.springframework.web.servlet.DispatcherServlet.render(DispatcherServlet.java:1211)
    org.springframework.web.servlet.DispatcherServlet.processDispatchResult(DispatcherServlet.java:1011)
    org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:955)
    org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:877)
    org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:966)
    org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:857)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:620)
    org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:842)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:727)
    org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:52)

【问题讨论】:

500错误有很多可能的原因,你可以有一个处理异常的方法,如果没有针对该特定异常的其他异常处理程序将被调用 @6ton 你能给我一个代码示例吗? @6ton 我给他写了一个表扬。它没有捕获异常。 能否添加异常发生时看到的堆栈跟踪 @6ton 包含堆栈跟踪。 【参考方案1】:

您的异常处理程序似乎返回了一个无效视图,这导致调用异常处理程序的代码中出现异常。见DispatcherServlet.processDispatchResult

在这种情况下,异常是从 DispatcherServlet 本身抛出的,处理它的唯一方法是通过web.xml

您需要修复异常处理程序以返回有效视图。使用@ExceptionHandler(Exception.class) 处理的其他答案是有效的,可用于处理控制器抛出的异常。

更新: 基于@Jack 的评论,异常是由控制器返回一个不存在的视图(而不是我最初认为的异常处理程序)引起的。一旦你返回一个视图,容器调用视图解析器来渲染它,如果抛出异常,异常处理程序将不会被调用(参见this & this)——简单的解决方案是在 web.xml 中处理它。 xml。

另一种选择是覆盖视图resolver.resolveViewName,假设您正在使用InternalResourceViewResolver,它可能如下所示:

public class CustomViewResolver extends InternalResourceViewResolver 
    @Override
    public View resolveViewName(String viewName,
                            Locale locale) throws Exception 
        try
            return super.resolveViewName();
         catch (Exception ex) 
            //log
            return new InternalResourceView("urlForErrorView");
       
    

【讨论】:

异常处理程序正在返回一个有效的视图。控制器返回无效的视图名称(在丹尼尔的情况下)“用户”。我使用了 @ExceptionHandler(Exception.class) 但它不处理异常。请给我一个示例代码,以确保我以正确的方式实现它。【参考方案2】:

似乎控制器建议无法处理此类异常。我在web.xml 中添加了以下代码来处理此异常。如果您有更好的解决方案,请告诉我。谢谢。

<error-page>
    <error-code>500</error-code>
    <location>/errorhandler</location>
</error-page>

【讨论】:

以上是关于如何让一个由 ControllerAdvice 注释的类来处理不同类型的异常?的主要内容,如果未能解决你的问题,请参考以下文章

SpringBoot 中 @ControllerAdvice 注解

SpringBoot 中 @ControllerAdvice 注解

SpringBoot 中 @ControllerAdvice 注解

如何在同一个 SpringMvc 应用程序中有两个 ControllerAdvice

 ControllerAdvice 是如何起作用的

玩转SpringBoot让错误处理重新由web服务器接管