前言
简单回顾下异常分类:
- 非受检异常:运行时异常
- 受检异常:编译时异常
异常的处理通常有两种方式:
- 第一种就是在当前类使用try-catch捕获异常并直接处理
- 另一种就是不在当前类处理,使用throws抛给上一级处理
如果在每个异常处都进行try-catch处理的话,无疑会造成代码的臃肿,且不够美观。
所以,一般来说除非的必须在在当前类处理的异常,否则都会向上抛出,直到抛到controller层为止,在controller层进行统一的异常处理(不处理的话,再往上抛就抛到前端去了)。
统一进行异常处理的常见方式有两种:1. 使用注解 或者 2. 使用(AOP)拦截器
controller类示例:
```java
@RestController
public class ErrorTestController {
@Autowired
ErrorService errorService;
@GetMapping("/test")
public ReturnResult toTest(){
HashMap<String, String> map = new HashMap<>();
map.put("data", "这里是返回的数据");
return ReturnResult.success(map);
}
@GetMapping("/test1")
public ReturnResult toTest1(){
errorService.error1();
return ReturnResult.success();
}
@GetMapping("/test2")
public ReturnResult toTest2(){
errorService.error2();
return ReturnResult.success();
}
}
同时在service类中模拟异常:
@Service
public class ErrorService {
public void error1() throws NullPointerException {
String[] str = null;
String s = str[0];
}
public void error2() throws ArithmeticException {
int i = 9 / 0;
}
}
AOP处理异常
使用AOP切controller类,进行统一的异常处理。
AOP切面类:
import email.entity.ReturnResult;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
import java.util.HashMap;
@Aspect
@Component
public class AOPCatch {
private final Logger logger = LoggerFactory.getLogger(AOPCatch.class);
@Around("execution(public * email.controller.*.*(..))")
public ReturnResult catchException(ProceedingJoinPoint joinPoint) {
try {
Object result = joinPoint.proceed();
// 没有异常,直接将接收到的被切方法的返回值原样返回
return (ReturnResult) result;
} catch (Throwable e) {
HashMap<String, String> map = new HashMap<>();
String className = joinPoint.getTarget().getClass().getName();
String methodName = joinPoint.getSignature().getName();
logger.error(className + ":" + methodName + ":" + e);
map.put("data", className + ":" + methodName + ":" + e);
ReturnResult error = ReturnResult.error(map);
// 有异常,则返回错误信息
return error;
}
}
}
注解处理异常
使用@ControllerAdvice
或者@RestControllerAdvice
(声明全局异常),然后配合@ExceptionHandler
注解使用
@RestControllerAdvice
注解就相当于@ControllerAdvice
+@ResponseBody
注解,标识该类中接收的返回信息的(json类型的)数据。
创建一个统一的异常处理类:
package email.servive;
import email.entity.ReturnResult;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import java.util.HashMap;
@RestControllerAdvice
//@ControllerAdvice
//@ResponseBody
public class GlobalExceptionHandler {
private final Logger logger = LoggerFactory.getLogger(AOPCatch.class);
// 全局异常处理
@ExceptionHandler(Exception.class)
public ReturnResult error(Exception e){
e.printStackTrace();
logger.error(e.getMessage());
return ReturnResult.error();
}
// 指定异常处理
@ExceptionHandler(NullPointerException.class)
public ReturnResult error(NullPointerException e){
e.printStackTrace();
logger.error(e.getMessage());
HashMap<String, String> map = new HashMap<>();
map.put("data", "空指针异常");
return ReturnResult.error(map);
}
// 自定义异常处理,针对于业务需求自定义的
/*
@ExceptionHandler(MyException.class)
public ReturnResult error(MyException e){
e.printStackTrace();
return ReturnResult.error();
}
*/
}
版权声明
本文依据
CC-BY-NC-SA 4.0
许可协议授权,请您在转载时注明文章来源为
Z次元
,若本文涉及转载第三方内容,请您一同注明。
知识积累
人非生而知之者,孰能无惑?惑而不从师,其为惑也,终不解矣。
评论区
发表评论
这里还没有评论哦
快来发一条评论抢占前排吧
前言
AOP处理异常
注解处理异常