一套标准的基础校验,可以将校验注解和附带错误信息添加到请求入参上即可完成校验,可以去除简单的校验代码,节省一定的时间和代码量
org.springframework.boot spring-boot-starter-validation
或直接使用以下即可,web 已包含
org.springframework.boot spring-boot-starter-web 2.0.5.RELEASE
都有也无所谓 只有一个第一个加载的生效
javax.validation validation-api 1.1.0.Final
org.hibernate hibernate-validator 5.4.1.Final
定义示例 controller
这里的 @valid 说明后面定义的实体需要进入校验,必须要加,否则实体内定义了校验注解不会生效
实体定义
import org.hibernate.validator.constraints.Range;import javax.validation.constraints.Max;
import javax.validation.constraints.Min;
import javax.validation.constraints.NotEmpty;
import javax.validation.constraints.NotNull;@Data
public class TestParams {@NotNull(message = "名称不能为空")private String name;@NotEmpty(message = "id 不能为空")private String idCard;@Min(value = 10, message = "最低不能低于10")private Integer number;@Max(value = 20, message = "最高不能高于20")private int maxNum;@Range(message = "区间只能位于{min}到{max}", min = 2, max = 12)private int rangeArea;}
@NotNull 不能为null但是可以为空
@NotEmpty 不能为空也不能为null
@Min 最小值
@Max 最大值
@Range 区间
通过上面的配置可以省去类似如下代码:
实际请求测试
{"name":"nnnnmamim","number":10,"maxNum":10,"rangeArea":3
}
返回的错误内容:
显然这样的内容是不可读的,需要整理
第一步可以如下修改添加
来源包
import org.springframework.validation.BindingResult;
结果如下:(这里我直接把错误返回了 实际应该将错误信息放到我们的返回体里面)
显然完成了校验
但是这样的处理未免太过麻烦,可以做全局异常捕获处理,但是我们要先知道这一类异常的类型才行
这种异常的异常类型如下:
MethodArgumentNotValidException
来源包:
import org.springframework.web.bind.MethodArgumentNotValidException;
具体捕捉可以如下:
spring-boot 中的全局异常处理:
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;@ControllerAdvice
@Slf4j
public class GlobalExceptionHandler {@ResponseBody@ExceptionHandler(value = MethodArgumentNotValidException.class)public Response handleValidException(HttpServletRequest request, MethodArgumentNotValidException e) {BindingResult bindingResult = e.getBindingResult();StringBuilder message = new StringBuilder();if (bindingResult.hasErrors()) {List allErrors = bindingResult.getAllErrors();for (ObjectError allError : allErrors) {message.append(allError.getDefaultMessage()).append(", ");}}return Response.failMessage(message.toString());}}
@ControllerAdvice
为对controller 的增强代码
@ResponseBody
json 返回体
@ExceptionHandler
指定不同异常类型的处理方式 针对不同的异常上面的特殊处理可以有多个
其中的 Response 为我自己定义的接口返回对象,Response.failMessage 会将失败信息放到的对象 message 属性中一并返回
最终测试结果:
此时在 Controller 中如下,其他不变 入参将不再需要 BindingResult
在 spring 以及 springMVC 中的另一种全局异常可以如下参考:
定义某一个类 实现接口 HandlerExceptionResolver 方法 resolveException
import lombok.extern.slf4j.Slf4j;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Component;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.servlet.HandlerExceptionResolver;
import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;@Slf4j
@Component
public class MyGlobalExceptionResolver implements HandlerExceptionResolver {@Overridepublic ModelAndView resolveException(HttpServletRequest request,HttpServletResponse response, Object handler, Exception ex) {log.error("全局异常", ex);ModelAndView modelAndView = new ModelAndView();// 转换异常if(ex instanceof MethodArgumentNotValidException) {modelAndView.setStatus(HttpStatus.OK);modelAndView.addObject("message", ex.getMessage());} else {// 其他所有异常modelAndView.setStatus(HttpStatus.INTERNAL_SERVER_ERROR);modelAndView.addObject("code", HttpStatus.INTERNAL_SERVER_ERROR);modelAndView.addObject("message", ex.getMessage());}response.setHeader("content-type", "application/json;charset=UTF-8");return modelAndView;}
}
下一篇:mysql基础操作2