@ExceptionHandler响应在Tomcat中不起作用

时间:2016-06-02 13:20:09

标签: java spring maven tomcat exception-handling

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

这是我的控制器:

@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;}
}

我正在使用tomcat 7插件为maven运行和调试应用程序,所以当我调试时,我可以看到@ExceptionHandler正在触发并执行return语句。但是,我收到的不是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)

如果我返回一个视图或者ModelView正在运行,但每当我尝试返回ResponseBody或ResponseEntity时,我仍然会遇到同样的错误。

最后,如果不是使用tomcat,我使用SpringBoot来运行应用程序,我没有收到错误,而且我的响应正如我应该收到的那样:

{"error":"E888"}

为什么不使用Tomcat?

感谢。

2 个答案:

答案 0 :(得分:0)

原因是Spring处理异常的方式。在没有Boot @ExceptionHandler的普通Spring应用程序中,exceptionResolver使用它,它适用于浏览器用户,换句话说,当异常到达时,它被认为是渲染页面。您的请求流不会通过contenNegotiationResolver,因此您将继续获得异常。在Spring启动中,请求的生命周期的fow被认为是浏览器用户的管理异常而不是浏览器用户之类的应用程序,因此它可以工作,我可以建议参考官方文档了解更多细节。

更新以回复评论。

在Spring启动中,您可以通过简单的spring mvc应用程序以非常不同的方式管理异常,我建议在链接spring boot doc处查看文档。 如果你想使用普通的弹簧网络应用程序,我可以建议使用一个方面。您可以使用普通的@ExceptionHandler用法并返回一个ModelView实现来回答@Controller类中没有@ResponseBody的普通页面请求,并创建一个方面来管理带有注释的方法的异常@ResponseBody并返回了ResponseEntity。这个方面可能是下面的想法,注意在这个示例中我开发了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
public class HelloController {


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

我希望它可以帮到你

答案 1 :(得分:0)

@ExceptionHandler方法没有@ResponseBody返回类型,只能返回转换为视图的ModelViewString

  

返回类型可以是String,它被解释为视图名称   或者是ModelAndView对象。

请点击此处了解详情:Spring @ExceptionHandler does not work with @ResponseBody