要想解决测试中存在的问题,我们需要对程序中可能出现的异常进行捕获,通常有两种处理方式:
A. 在方法中加入 try…catch 进行异常捕获
形式如下:
如果采用这种方式,虽然可以解决,但是存在弊端,需要我们在保存其他业务数据时,也需要在其他方法中加上try…catch进行处理,代码冗余,不通用。
B. 使用异常处理器进行全局异常捕获
采用这种方式来实现,我们只需要在项目中定义一个通用的全局异常处理器,就可以解决本项目的所有异常。
在项目中自定义一个全局异常处理器,在异常处理器上加上注解 @ControllerAdvice,可以通过属性annotations指定拦截哪一类的Controller方法。 并在异常处理器的方法上加上注解 @ExceptionHandler 来指定拦截的是那一类型的异常。
异常处理方法逻辑:
- 指定捕获的异常类型为 SQLIntegrityConstraintViolationException
- 解析异常的提示信息, 获取出是那个值违背了唯一约束
- 组装错误信息并返回
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.bind.annotation.RestController;
import java.sql.SQLIntegrityConstraintViolationException;
/**
* 全局异常处理
*/
@ControllerAdvice(annotations = {RestController.class, Controller.class})
@ResponseBody
@Slf4j
public class GlobalExceptionHandler {
/**
* 异常处理方法
* @return
*/
@ExceptionHandler(SQLIntegrityConstraintViolationException.class)
public R<String> exceptionHandler(SQLIntegrityConstraintViolationException ex){
log.error(ex.getMessage());
if(ex.getMessage().contains("Duplicate entry")){
String[] split = ex.getMessage().split(" ");
String msg = split[2] + "已存在";
return R.error(msg);
}
return R.error("未知错误");
}
}
代码说明:
上述的全局异常处理器上使用了的两个注解 @ControllerAdvice , @ResponseBody , 他们的作用分别为:
@ControllerAdvice
: 指定拦截那些类型的控制器;
@ResponseBody
: 将方法的返回值 R 对象转换为json格式的数据, 响应给页面;
annotations
属性指定只对带有@RestController或@Controller注解的类中的方法进行异常处理。
@ExceptionHandler(SQLIntegrityConstraintViolationException.class)
:使用@ExceptionHandler注解标识这是一个异常处理方法,处理SQLIntegrityConstraintViolationException异常。
全局异常处理器编写完毕之后,我们需要将项目重启, 完毕之后直接访问管理系统首页, 点击 “员工管理” 页面中的 “添加员工” 按钮。当我们在测试中,添加用户时, 输入了一个已存在的用户名时,前端界面出现如下错误提示信息:
@ControllerAdvice
注解可以用于类级别,用于标记一个类为全局异常处理器。@RestController
、@Controller
注解一起使用,用于指定全局异常处理器只处理带有 @RestController
或 @Controller
注解的控制器类中抛出的异常。@ControllerAdvice
注解标记的类中定义异常处理方法,可以统一处理应用程序中的各种异常。@ExceptionHandler
注解进行标记,以指定处理的异常类型。@ControllerAdvice
注解可以限定作用的范围,即指定哪些控制器类的异常会被当前的全局异常处理器处理。annotations
、basePackages
、basePackageClasses
等属性来指定作用的范围。@ControllerAdvice
还可以用于其他方面,比如全局数据绑定、全局数据预处理等。@ExceptionHandler
注解可以用于方法级别,用于标记一个方法为异常处理方法。总的来说,@ExceptionHandler
注解提供了一种在控制器中处理异常的机制,能够根据不同类型的异常来执行不同的异常处理逻辑,使代码更加清晰和易于维护。