java - @ExceptionHandler 响应在 Tomcat 中不起作用

标签 java spring maven tomcat exception

我正致力于使用 Spring MVC 处理异常。我正在使用@ControllerAdvice 和@ExceptionHandler。为了返回一个 JSON,我尝试了两种情况:@ResponseBody 和 ResponseEntity。

这是我的 Controller :

@ControllerAdvice
public class GlobalExceptionController {

@ExceptionHandler(CustomGenericException.class)
@ResponseBody
public ErrorResource handleCustomException(CustomGenericException ex) {

        ErrorResource errorResource=new ErrorResource("Example 1");
    errorResource.error=ex.getErrCode();

    return errorResource;

}

@ExceptionHandler(Exception.class)
public ResponseEntity<ErrorResource> handleAllException(Exception ex) {

    ErrorResource errorResource=new ErrorResource("Example 2");
    errorResource.error=ex.getMessage();

    return new ResponseEntity<ErrorResource>(errorResource,HttpStatus.NOT_FOUND);

}

}
class ErrorResource {
public String error;

public ErrorResource(String a ){error=a;}
}

我正在使用 maven 的 tomcat 7 插件来运行和调试应用程序,因此在调试时我可以看到 @ExceptionHandler 正在触发并执行返回语句。但是我收到的不是 JSON,而是 HTTP 500 错误:

org.springframework.web.util.NestedServletException: Request processing failed; nested exception is com.mkyong.web.exception.CustomGenericException
org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:979)
org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:858)
javax.servlet.http.HttpServlet.service(HttpServlet.java:621)
org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:843)
javax.servlet.http.HttpServlet.service(HttpServlet.java:728)
org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:51)

如果我返回一个 View 或一个 ModelView 正在工作,但每当我尝试返回一个 ResponseBody 或 ResponseEntity 时,我仍然会遇到同样的错误。

最后,如果我不使用 tomcat,而是使用 SpringBoot 来运行该应用程序,我不会收到错误消息,并且会收到我应该收到的响应:

{"error":"E888"}

为什么不使用 Tomcat?

谢谢。

最佳答案

原因是Spring处理异常的方式。在没有引导的普通 Spring 应用程序中,exceptionResolver 使用 @ExceptionHandler 并且它适用于浏览器用户,或者换句话说,当异常到达时它被认为是渲染页面。您的请求流不会通过 contenNegotiationResolver,因此您会继续收到异常。在 Spring boot 中,请求的生命周期流程被认为是为浏览器用户和非浏览器用户之类的应用程序管理异常,因此它有效,我建议引用官方文档以获取更多详细信息。

更新以回答评论。

在 Spring boot 中,您以非常不同的方式管理异常,尊重普通 spring mvc 应用程序的许多细节,我建议在链接 spring boot doc 上查看文档. 如果你想使用一个普通的 spring web 应用程序,我可以建议使用一个方面。您可以使用普通的 @ExceptionHandler 用法并返回一个 ModelView 实现以响应 @Controller 类中没有 @ResponseBody 的普通页面请求并创建一个方面,它管理用 @ResponseBody 注释的方法的异常并返回 ResponseEntity。 Thew 方面可以像下面这样思考,请注意,在这个示例中,我开发了 poincut 表达式,用于拦截返回 com.springapp.mvc 包下的 ResponseEntity 的方法:

看点:

@Aspect
@Component
public class GlobalExceptionRestAOPController {


    private ObjectMapper objectMapper = new ObjectMapper();

    @Around("execution(org.springframework.http.ResponseEntity com.springapp.mvc..*(..))")
    public Object manageException(ProceedingJoinPoint pjp) throws Throwable{
        Object result;
        try{
            result = pjp.proceed();
        } catch (Exception e){
            // log it.
                       result = ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).contentType(MediaType.APPLICATION_JSON).body(objectMapper.writeValueAsString(new ExceptionReport(e.toString())));

        }

        return result;
    }
}

class ExceptionReport{
    private final String error;

    public ExceptionReport(String error) {
        this.error = error;
    }

    public String getError() {
        return error;
    }
}

不要忘记在你的配置中插入

xml 样式:

<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">

    <aop:aspectj-autoproxy/>

    <context:component-scan base-package="com.springapp.mvc"/>

    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/WEB-INF/pages/"/>
        <property name="suffix" value=".jsp"/>
    </bean>

...
</beans>    

Java 风格:

@Configuration
@EnableAspectJAutoProxy
...
class AspectConfiguration{
...
}

Controller :

@Controller
public class HelloController {


    @RequestMapping(value = "/jsonHelloEx", method = RequestMethod.GET, produces = "application/json")
    public ResponseEntity printWelcomeWithException() {
        String nullableString = null;
        nullableString.equals("");
        return ResponseEntity.ok("Hello");
    }
}

希望对你有帮助

关于java - @ExceptionHandler 响应在 Tomcat 中不起作用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37593087/

相关文章:

java - 将 HttpServletRequest 转换为 JSON 作为 "UTF-8"

Spring 数据 JPA : How to create "delete by multiple properties" method?

java.lang.ClassNotFoundException : org. springframework.cglib.core.CodeGenerationException

java - Java中不同类型的串联

java - 在 JPA 中实现互斥关系的最佳方法是什么?

java - 如何运行tomcat7-maven-plugin,服务器代码更改将立即更新

java - maven package phase创建的war文件和maven war plugin war :war goal创建的war文件有区别吗

git - 使用 GitLab CI 在一个存储库中运行多个 Maven 项目

java - 我的背景图像对于 Canvas 来说太大了

java - 使用 spring batch 并行执行作业