java mvc异常捕获_spring mvc统一异常处理(@ControllerAdvice + @ExceptionHandler)

v一、前言

项目刚刚开发的时候,并没有做好充足的准备。开发到一定程度的时候才会想到还有一些问题没有解决。就比如今天我要说的一个问题:异常的处理。写程序的时候一般都会通过try...catch...finally对异常进行处理,但是我们真的能在写程序的时候处理掉所有可能发生的异常吗? 以及发生异常的时候执行什么逻辑,返回什么提示信息,跳转到什么页面,这些都是要考虑到的。

v二、基于@ControllerAdvice(加强的控制器)的异常处理

@ControllerAdvice注解内部使用@ExceptionHandler、@InitBinder、@ModelAttribute注解的方法应用到所有的 @RequestMapping注解的方法。本例子中使用ExceptionHandler应用到所有@RequestMapping注解的方法,处理发生的异常。

示例代码:

f873f8c2c2319b9d574edb111fa9a461.gif

importorg.slf4j.Logger;importorg.slf4j.LoggerFactory;importorg.springframework.http.HttpHeaders;importorg.springframework.http.HttpStatus;importorg.springframework.http.ResponseEntity;importorg.springframework.util.StringUtils;importorg.springframework.web.bind.annotation.ControllerAdvice;importorg.springframework.web.bind.annotation.ExceptionHandler;importorg.springframework.web.bind.annotation.ResponseBody;importcom.hjz.exception.ServiceException;importcom.hjz.exception.utils.ExceptionUtils;

@ResponseBodypublic classExceptionAdvice {private static final Logger LOGGER = LoggerFactory.getLogger(ExceptionAdvice.class);/*** 拦截web层异常,记录异常日志,并返回友好信息到前端

* 目前只拦截Exception,是否要拦截Error需再做考虑

*

*@parame 异常对象

*@return异常提示*/@ExceptionHandler(Exception.class)public ResponseEntityhandleException(Exception e) {//不需要再记录ServiceException,因为在service异常切面中已经记录过

if (!(e instanceofServiceException)) {

LOGGER.error(ExceptionUtils.getExcTrace(e));

}

HttpHeaders headers= newHttpHeaders();

headers.set("Content-type", "text/plain;charset=UTF-8");

headers.add("icop-content-type", "exception");

String message= StringUtils.isEmpty(e.getMessage()) ? "系统异常!!": e.getMessage();return new ResponseEntity<>(message, headers, HttpStatus.OK);

}

}

f873f8c2c2319b9d574edb111fa9a461.gif

如果不起作用,请检查 spring-mvc的配置文件,是否有ControllerAdvice的如下配置

v三、基于AOP的异常处理

1.处理controller层的异常 WebExceptionAspect.java

f873f8c2c2319b9d574edb111fa9a461.gif

importorg.aspectj.lang.annotation.AfterThrowing;importorg.aspectj.lang.annotation.Aspect;importorg.aspectj.lang.annotation.Pointcut;importorg.slf4j.Logger;importorg.slf4j.LoggerFactory;importorg.springframework.stereotype.Component;importorg.springframework.util.StringUtils;importorg.springframework.web.context.request.RequestContextHolder;importorg.springframework.web.context.request.ServletRequestAttributes;importcom.hjz.exception.ServiceException;importcom.hjz.exception.utils.ExceptionUtils;importjavax.servlet.http.HttpServletResponse;importjava.io.IOException;importjava.io.PrintWriter;/*** web异常切面

* 默认spring aop不会拦截controller层,使用该类需要在spring公共配置文件中注入改bean,

* 另外需要配置*/@Aspectpublic classWebExceptionAspect {private static final Logger LOGGER = LoggerFactory.getLogger(WebExceptionAspect.class);

@Pointcut("@annotation(org.springframework.web.bind.annotation.RequestMapping)")private voidwebPointcut() {}/*** 拦截web层异常,记录异常日志,并返回友好信息到前端

* 目前只拦截Exception,是否要拦截Error需再做考虑

*

*@parame 异常对象*/@AfterThrowing(pointcut= "webPointcut()", throwing = "e")public voidhandleThrowing(Exception e) {//不需要再记录ServiceException,因为在service异常切面中已经记录过

if (!(e instanceofServiceException)) {

LOGGER.error(ExceptionUtils.getExcTrace(e));

}

String errorMsg= StringUtils.isEmpty(e.getMessage()) ? "系统异常": e.getMessage();

writeContent(errorMsg);

}/*** 将内容输出到浏览器

*

*@paramcontent 输出内容*/

private voidwriteContent(String content) {

HttpServletResponse response=((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getResponse();

response.reset();

response.setCharacterEncoding("UTF-8");

response.setHeader("Content-Type", "text/plain;charset=UTF-8");

response.setHeader("icop-content-type", "exception");

PrintWriter writer= null;try{

writer=response.getWriter();

}catch(IOException e) {

e.printStackTrace();

}

writer.print(content);

writer.flush();

writer.close();

}

}

f873f8c2c2319b9d574edb111fa9a461.gif

2.处理service层的异常ServiceExceptionAspect .java

f873f8c2c2319b9d574edb111fa9a461.gif

importorg.aspectj.lang.JoinPoint;importorg.aspectj.lang.annotation.AfterThrowing;importorg.aspectj.lang.annotation.Aspect;importorg.aspectj.lang.annotation.Pointcut;importorg.slf4j.Logger;importorg.slf4j.LoggerFactory;importorg.springframework.stereotype.Component;importorg.springframework.util.StringUtils;importcom.hjz.exception.ServiceException;importcom.hjz.exception.utils.ExceptionUtils;

@Aspectpublic classServiceExceptionAspect {private static final Logger LOGGER = LoggerFactory.getLogger(ServiceExceptionAspect.class);/*** @within(org.springframework.stereotype.Service),拦截带有 @Service 注解的类的所有方法

* @annotation(org.springframework.web.bind.annotation.RequestMapping),拦截带有@RquestMapping的注解方法*/@Pointcut("@within(org.springframework.stereotype.Service) && execution(public * *(..))")private voidservicePointcut() {}/*** 拦截service层异常,记录异常日志,并设置对应的异常信息

* 目前只拦截Exception,是否要拦截Error需再做考虑

*

*@parame 异常对象*/@AfterThrowing(pointcut= "servicePointcut()", throwing = "e")public voidhandle(JoinPoint point, Exception e) {

LOGGER.error(ExceptionUtils.getExcTrace(e));

String signature=point.getSignature().toString();

String errorMsg= getMessage(signature) == null ? (StringUtils.isEmpty(e.getMessage()) ? "服务异常": e.getMessage()) : getMessage(signature);throw newServiceException(errorMsg, e);

}/*** 获取方法签名对应的提示消息

*

*@paramsignature 方法签名

*@return提示消息*/

privateString getMessage(String signature) {return null;

}

}

f873f8c2c2319b9d574edb111fa9a461.gif

3.使用方式,在spring的公共配置文件中加入如下配置:

或者 自定义一个 注册类,ServiceExceptionAspect.java和WebExceptionAspect.java都加入@Component注解

f873f8c2c2319b9d574edb111fa9a461.gif

importorg.springframework.context.annotation.ComponentScan;importorg.springframework.context.annotation.Configuration;importorg.springframework.context.annotation.EnableAspectJAutoProxy;/*** 异常相关bean注册类*/@Configuration

@EnableAspectJAutoProxy

@ComponentScan("com.hjz.exception.aspect")public classExceptionConfig {

}

f873f8c2c2319b9d574edb111fa9a461.gif

注:spring 公共配置文件中的配置 改成 ,如果controller层的异常无法拦截,请将配置换到springmvc的配置文件中,原因请见(

f873f8c2c2319b9d574edb111fa9a461.gif

@Aspect

@Componentpublic classWebExceptionAspect {

..........

}

@Aspect

@Componentpublic classServiceExceptionAspect {

.........

}

f873f8c2c2319b9d574edb111fa9a461.gif

v四、疑惑

@within(org.springframework.stereotype.Service),拦截带有 @Service 注解的类的所有方法

@annotation(org.springframework.web.bind.annotation.RequestMapping),拦截带有@RquestMapping的注解方法

v五、测试

分别编写controller层和service层的异常测试类。这个很简单,在方法里简单的抛一下异常就可以了。最后验证一下,异常发生的时候有没有 执行 @AfterThrowing对应的方法就好了。具体还是看我写的demo吧,嘿嘿嘿!!!

http://www.cnblogs.com/hujunzheng/p/6255463.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值