小编给大家分享一下java怎么实现统一异常处理,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!
对于Dao层 和Service产生的异常要一直网上抛,直至Controller层,但是对于controller层不能处理的异常也不能直接抛给前端。
为什么不能在service处理异常?
答:Service 层往往涉及数据库事务,出现异常同样不适合捕获,否则事务无法自动回滚。此外 Service 层涉及业务逻辑,有些业务逻辑执行中遇到业务异常,可能需要在异常后转入分支业务流程。如果业务异常都被框架捕获了,业务功能就会不正常。【引用:极客时间的Java业务开发常见错误100例】
实现统一异常处理:
在spring框架下实现一个异常处理的类,用 @RestControllerAdvice + @ExceptionHandler
进行修饰:
即@RestControllerAdvice默认会拦截 controller类上抛出的不能处理的异常
一个全局异常处理类需要处理三类异常: 1.业务类异常,2.运行时异常 ,3.Error
1.运行时异常
/**
* @创建人: liup
* @创建时间: 2021/6/18
* @描述 全局异常捕获处理类
*/
@RestControllerAdvice
@Slf4j
public class GlobalExceptionHandler {
/**
* @Author: liup
* @date: 2021/6/18 14:34
方法实现说明: 拦截运行时异常
*/
@ExceptionHandler(value = RuntimeException.class)
public R runtimeExceptionHandle(RuntimeException e){
log.error("捕捉到运行时异常",e);
return R.failed("未知错误");
}
}
目前仅是拦截运行时异常
R 是返回的消息体:
那如果不使用GlobalExceptionHandler,会报出什么错误呢?
这个错误是在service层抛出的,当从redis 通过key获取一个已删除的value时,redis返回的是null,但是我没有判断这个value是否为null,就将其打印出来:
log.info(authInfoVo.toString());
注意:这是要返回给前端的,msg的内容,是对用户十分不友好的。
2.Error
RuntimeException只是异常中的一个类,不能包含所有的异常体系,还有一大类是叫Error(系统级异常),所以需要有一个兜底的异常捕获:
/**
* @Author: liup
* @date: 2021/6/18 15:01
方法实现说明: 捕获系统级异常
*/
@ExceptionHandler(value = Throwable.class)
public R throwableHandle(Throwable th){
log.error("捕捉到Throwable异常",th);
return R.failed("系统异常");
}
和上面那个运行时异常同时存在 。
3.业务类异常
【自己定义的异常】
首先创建业务异常类
/**
* @创建人: liup
* @创建时间: 2021/6/18
* @描述 业务类异常
*/
public class BusinessException extends RuntimeException{
@Getter
private final String code;
/**
* @Author: liup
* @date: 2021/6/18 15:10
方法实现说明: 根据消息码【可用枚举类】 构造业务类异常
*/
public BusinessException(String code) {
this.code = code;
}
/**
* @Author: liup
* @date: 2021/6/18 15:08
方法实现说明: 自定义消息体构造业务类异常
*/
public BusinessException(String code,String message) {
super(message);
this.code = code;
}
/**
* @Author: liup
* @date: 2021/6/18 15:09
方法实现说明: 根据异常 构造业务类异常
*/
public BusinessException(String code,Throwable cause) {
super(cause);
this.code = code;
}
}
三种异常拦截同时存在;
/**
* @创建人: liup
* @创建时间: 2021/6/18
* @描述 全局异常捕获处理类
*/
@RestControllerAdvice
@Slf4j
public class GlobalExceptionHandler {
/**
* @Author: liup
* @date: 2021/6/18 15:14
方法实现说明: 拦截业务类异常
*/
@ExceptionHandler(value = BusinessException.class)
public R businessExceptionHandle(BusinessException e){
log.error("捕获业务类异常:",e);
return R.failed("业务类异常:"+e.getMessage());
}
/**
* @Author: liup
* @date: 2021/6/18 14:34
方法实现说明: 拦截运行时异常
// */
@ExceptionHandler(value = RuntimeException.class)
public R runtimeExceptionHandle(RuntimeException e){
log.error("捕捉到运行时异常",e);
return R.failed("未知错误:");
}
/**
* @Author: liup
* @date: 2021/6/18 15:01
方法实现说明: 捕获系统级异常
*/
@ExceptionHandler(value = Throwable.class)
public R throwableHandle(Throwable th){
log.error("捕捉到Throwable异常",th);
return R.failed("系统异常");
}
}
以上是对前端友好,但是在服务器上,不是容易定位错误,
但是若是在参数上添加上HttpServletRequest req, HandlerMethod method,就很容易定位到错误
private static int GENERIC_SERVER_ERROR_CODE = 2000;
private static String GENERIC_SERVER_ERROR_MESSAGE = "服务器忙,请稍后再试";
@ExceptionHandler
public R handle(HttpServletRequest req, HandlerMethod method, Exception ex) {
if (ex instanceof BusinessException) {
BusinessException exception = (BusinessException) ex;
log.warn(String.format("访问 %s -> %s 出现业务异常!", req.getRequestURI(), method.toString()), ex);
return R.failed(GENERIC_SERVER_ERROR_MESSAGE);
} else if (ex instanceof RuntimeException){
log.error(String.format("访问 %s -> %s 出现运行时异常!", req.getRequestURI(), method.toString()), ex);
return R.failed(GENERIC_SERVER_ERROR_MESSAGE);
}
else {
log.error(String.format("访问 %s -> %s 出现系统异常!", req.getRequestURI(), method.toString()), ex);
return R.failed(GENERIC_SERVER_ERROR_MESSAGE);
}
}
以上是“java怎么实现统一异常处理”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注天达云行业资讯频道!