Spring 4 异常处理

时间:2022-08-26 00:10:02

异常与HTTP状态码的映射(@ResponseStatus)

Spring默认会将自身抛出的异常自动映射到合适的状态码,如下是一些示例:

Spring 4 异常处理

举个例子,当后端抛出如下异常(TypeMismatchException异常,往方法传参时,类型不匹配):

org.springframework.beans.TypeMismatchException: Failed to convert value of type 'java.lang.String' to required type 'long'; nested exception is java.lang.NumberFormatException: For input string: "2l"
    at org.springframework.beans.TypeConverterSupport.doConvert(TypeConverterSupport.java:77)
    at org.springframework.beans.TypeConverterSupport.convertIfNecessary(TypeConverterSupport.java:47)
    at org.springframework.validation.DataBinder.convertIfNecessary(DataBinder.java:603)
...

前台返回400状态码:

Spring 4 异常处理

除了以上异常,对于其它异常以及我们业务自己抛出的异常,如果没有明确绑定Http状态码,响应默认都会带有500状态码。

当然,除了这些默认机制,我们也可以将自定义异常绑定特点的Http状态码,通过@ResponseStatus注解可实现,如下示例:

 定义一个异常,通过@ResponseStatus注解绑定400状态码:

@ResponseStatus(value = HttpStatus.NOT_FOUND)
public class MyException extends RuntimeException
{

}

然后再controller抛出自定义异常throw new MyException();

访问controller,发现响应确实返回了400状态码。

控制器中的异常处理方法(@ExceptionHandler)

异常处理方法能处理同一个controller中所有方法抛出的异常,如下示例:

我们在controller下添加了一个MyException异常的处理方法,直接返回到body。

    @ExceptionHandler(MyException.class)
    @ResponseBody
    public String handleException(){
        return "handle by ExceptionHandler.";
    }

打开浏览器,观察结果:

Spring 4 异常处理

控制器通知(@ControllerAdvice)

异常处理方法只能处理同一个controller中抛出的异常,然而一个系统,肯定不止一个controller,总不可能在每个controller中都添加重复性的异常处理方法吧~~

那么对于多个controller,如何处理异常呢?使用@ControllerAdvice注解即可。

带有@ControllerAdvice注解的类,可以收到系统中所有Controller抛出的异常,如下示例:

@ControllerAdvice
public class DSSExceptionHandler extends BaseController
{

    /**
     * 处理controller抛出的异常
     *
     * @return
     */
    @ExceptionHandler(Exception.class)
    @ResponseBody
    public String handleException(HttpServletRequest request, Exception e)
    {
        logger.error("Request FAILD, URL = {} ", request.getRequestURI());
        logger.error(e.toString(), e);
        return gson.toJson(BaseController.FAILD);
    }

    /**
     * 处理controller抛出的异常
     *
     * @return
     */
    @ExceptionHandler(NumberFormatException.class)
    @ResponseBody
    public String handleNumberFormatException(HttpServletRequest request, NumberFormatException e)
    {
        logger.error("Request FAILD, URL = {} ", request.getRequestURI());
        logger.error(e.toString(), e);
        return gson.toJson(BaseController.FAILD);
    }

}

有一个点注意下,就是spring 扫描配置的时候,要包括该bean,我的配置如下,可参考:

spring-mvc.xml:

    <context:component-scan base-package="com.cetiti.epdc.dss" >
        <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>  
        <context:include-filter type="annotation" expression="org.springframework.web.bind.annotation.ControllerAdvice"/> 
    </context:component-scan> 

spring.xml

    <context:component-scan base-package="com.cetiti.epdc.dss">
        <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller" />
        <context:exclude-filter type="annotation" expression="org.springframework.web.bind.annotation.ControllerAdvice" />
    </context:component-scan>

另外,在上面的示例中,范围更小的异常,优先级更大,所以会调用handleNumberFormatException方法。

参考资料

spring in action 4