报错SQLIntegrityConstraintViolationException
案例
需求,添加用户到数据库
当我们在测试添加用户时, 输入了一个已存在的用户名,前端界面会出现错误提示信息:500,
而此时,服务端已经报错了, 报错信息如下:
出现上述的错误, 主要就是因为在 employee 表结构中,我们针对于username字段,建立了唯一索引,添加重复的username数据时,违背该约束,就会报错。但是此时前端提示的信息并不具体,用户并不知道是因为什么原因造成的该异常,我们需要给用户提示详细的错误信息 。
要想解决上述测试中的问题,我们需要对程序中可能出现的异常进行捕获处理,一般有两种方式:
A. 在 Controller方法中加入 try...catch 进行异常捕获 :
try {
EmployeeService.save(employee);
} catch (Exception e) {
e.printStackTrace();
return R.error(("新增员工失败"));
}
如果用这种方式可以解决问题,但也存在弊端——我们在保存其他业务数据时,也需要在Controller方法中加上try...catch进行处理,代码冗余
B. 使用异常处理器进行全局异常捕获
采用这种方式来实现,我们只需要在项目中定义一个通用的全局异常处理器,就可以解决项目中所有异常。
全局项目处理器
在项目中自定义一个全局异常处理器,在异常处理器上加上注解 @ControllerAdvice,可以通过属性annotations指定拦截哪一类的Controller方法。
并在异常处理器的方法上加上注解@ExceptionHandler 来指定拦截的是那一类型的异常。
@ControllerAdvice : 指定拦截那些类型的控制器;
@ResponseBody: 将方法的返回值 R 对象转换为json格式的数据, 响应给页面;
上述使用的两个注解, 也可以合并成为一个注解 @RestControllerAdvice
在service层查询用户是否存在,如果存在则抛出异常
//查询用户名是否存在
LambdaQueryWrapper<Employee> queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.eq(Employee::getUsername,employee.getUsername());
Integer count = employeeMapper.selectCount(queryWrapper);
if(count > 0){
throw new BusinessException("用户名["+employee.getUsername()+"]已经存在");
}
新建业务异常类
package com.alibaba.exception;
/**
* 业务异常
*/
public class BusinessException extends RuntimeException{
public BusinessException(String message){
super(message);
}
}
新建全局异常处理类
package com.alibaba.exception;
import com.alibaba.common.R;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
/**
* 全局异常处理类
*/
@RestControllerAdvice
@Slf4j
public class GlobalExceptionHandler {
/**
* 业务异常
* @param businessException
* @return
*/
@ExceptionHandler(BusinessException.class)
public R<String> businessException(BusinessException businessException){
log.error(businessException.getMessage());
return R.error(businessException.getMessage());
}
/**
* 未知异常
* @param exception
* @return
*/
@ExceptionHandler(Exception.class)
public R<String> exception(Exception exception){
log.error(exception.getMessage());
return R.error(exception.getMessage());
}
}
测试:
前端:
后台:
这样就使得代码更加优雅!