JSR303校验详解(基础校验,分组校验,自定义校验)结合 全局异常处理捕获校验异常


前言

这篇文章学习自尚硅谷
首先让我们来看看基本思路

 * 3、JSR303
 *   1)、给Bean添加校验注解:javax.validation.constraints,并定义自己的message提示
 *   2)、开启校验功能@Valid
 *      效果:校验错误以后会有默认的响应;
 *   3)、给校验的bean后紧跟一个BindingResult,就可以获取到校验的结果
 *   4)、分组校验(多场景的复杂校验)
 *         1)、	@NotBlank(message = "品牌名必须提交",groups = {AddGroup.class,UpdateGroup.class})
 *          给校验注解标注什么情况需要进行校验
 *         2)、@Validated({AddGroup.class})
 *         3)、默认没有指定分组的校验注解@NotBlank,在分组校验情况@Validated({AddGroup.class})下不生效,只会在@Validated生效;
 *
 *   5)、自定义校验
 *      1)、编写一个自定义的校验注解
 *      2)、编写一个自定义的校验器 ConstraintValidator
 *      3)、关联自定义的校验器和自定义的校验注解
         *      @Documented
         * @Constraint(validatedBy = { ListValueConstraintValidator.class【可以指定多个不同的校验器,适配不同类型的校验】 })
         * @Target({ METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER, TYPE_USE })
         * @Retention(RUNTIME)
         * public @interface ListValue {
 *
 * 4、统一的异常处理
 * @ControllerAdvice
 *  1)、编写异常处理类,使用@ControllerAdvice。
 *  2)、使用@ExceptionHandler标注方法可以处理的异常。
 */

0 学习准备工作

1 新建一个springboot项目

直接从新建一个项目开始学习
这个你肯定会的

2 导入所用依赖

所用依赖给出

    <dependencies>
<!--        web mvc-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <!--        mysql链接-->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>
<!--        lombok简化插件-->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
<!--        test测试-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <!--        myabtis-plus依赖-->
        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>mybatis-plus-boot-starter</artifactId>
            <version>3.2.0</version>
        </dependency>
    </dependencies>

3 创建我们测试用的数据库(sql语句给出)

在这里插入图片描述

/*
Navicat MySQL Data Transfer

Source Server         : localhost_3306
Source Server Version : 50726
Source Host           : localhost:3306
Source Database       : jsr303

Target Server Type    : MYSQL
Target Server Version : 50726
File Encoding         : 65001

Date: 2022-05-12 21:13:27
*/

SET FOREIGN_KEY_CHECKS=0;

-- ----------------------------
-- Table structure for test
-- ----------------------------
DROP TABLE IF EXISTS `test`;
CREATE TABLE `test` (
  `id` int(10) NOT NULL AUTO_INCREMENT,
  `name` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL COMMENT '姓名',
  `email` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL COMMENT '邮箱',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

-- ----------------------------
-- Records of test
-- ----------------------------
INSERT INTO `test` VALUES ('1', '孙悟空', '1');
INSERT INTO `test` VALUES ('2', '八戒', '2');
INSERT INTO `test` VALUES ('3', '唐僧', '3');

4 application.yml的配置文件

spring:
  datasource:
    driver-class-name: com.mysql.cj.jdbc.Driver
    username: root
    password: 123456
    url: jdbc:mysql://localhost:3306/jsr303?useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true&useSSL=false&serverTimezone=GMT%2B8
server:
  port: 8082
mybatis-plus:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

5 Mybatis-plus教程

可以简单看看我的这篇博客
Mybatis-plus快速上手

1 基础JSR303校验

我们先给后端加上简单的JSR303校验

0 思路:

  • JSR303基础校验
  • 1)、给Bean添加校验注解:javax.validation.constraints,并定义自己的message提示
  • 2)、开启校验功能@Valid
  •  效果:校验错误以后会有默认的响应;
    
  • 3)、给校验的bean后紧跟一个BindingResult,就可以获取到校验的结果

首先导入依赖

        <!--        校验注解-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-validation</artifactId>
            <version>2.6.7</version>
        </dependency>

1 我们这里给TestEntity 实体类来加上注解

@Data
@TableName("test")
public class TestEntity implements Serializable {

    //忽略掉 设置序列化和反序列化的serialVersionUID
    //想了解一下可以去这篇博客 https://blog.csdn.net/qq_45503106/article/details/107950914
    private static final long serialVersionUID = 1L;

    /**
     * test表主键ID
     */
    @TableId(type = IdType.AUTO)
    private Integer id;
    /**
     *  姓名
     */
    @NotEmpty(message = "姓名不能为空")
    private String name;

    /**
     * 邮箱
     */
    private String email;

}

在这里插入图片描述

2 错误解决 注解不生效问题

首先看看你导的是不是java.javax.validation.constraints.*;
在这里插入图片描述
如果不是引入一下依赖

        <!--        校验注解-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-validation</artifactId>
            <version>2.6.7</version>
        </dependency>

更新依赖引入后如果重启还不生效
maven
先clean 再 compile试试

3 controller层开启校验功能

代码如下

@PostMapping("/save")
    public String save(@Valid @RequestBody TestEntity testEntity, BindingResult result){
        if(result.hasErrors()){
            Map<String,String> map = new HashMap<>();
            //1、获取校验的错误结果
            result.getFieldErrors().forEach((item)->{
                //FieldError 获取到错误提示
                String message = item.getDefaultMessage();
                //获取错误的属性的名字
                String field = item.getField();
                map.put(field,message);
            });

            return map.toString();
        }else {
            testService.save(testEntity);
            return "ok";
        }

    }

在这里插入图片描述

2 统一异常处理

  • 统一的异常处理
  • @ControllerAdvice
  • 1)、编写异常处理类,使用@ControllerAdvice。
  • 2)、使用@ExceptionHandler标注方法可以处理的异常。

统一返回类编写
这里使用人人开源的模板

package com.example.jsr303.common.utils;

import org.apache.http.HttpStatus;

import java.util.HashMap;
import java.util.Map;

/**
 * 返回数据
 *
 * @author Mark sunlightcs@gmail.com
 */
public class R extends HashMap<String, Object> {
	private static final long serialVersionUID = 1L;

	public R() {
		put("code", 0);
		put("msg", "success");
	}

	public static R error() {
		return error(HttpStatus.SC_INTERNAL_SERVER_ERROR, "未知异常,请联系管理员");
	}

	public static R error(String msg) {
		return error(HttpStatus.SC_INTERNAL_SERVER_ERROR, msg);
	}

	public static R error(int code, String msg) {
		R r = new R();
		r.put("code", code);
		r.put("msg", msg);
		return r;
	}

	public static R ok(String msg) {
		R r = new R();
		r.put("msg", msg);
		return r;
	}

	public static R ok(Map<String, Object> map) {
		R r = new R();
		r.putAll(map);
		return r;
	}

	public static R ok() {
		return new R();
	}

	public R put(String key, Object value) {
		super.put(key, value);
		return this;
	}
}

全局枚举编写

package com.example.jsr303.common.exception;

/***
 * 错误码和错误信息定义类
 * 1. 错误码定义规则为5为数字
 * 2. 前两位表示业务场景,最后三位表示错误码。例如:100001。10:通用 001:系统未知异常
 * 3. 维护错误码后需要维护错误描述,将他们定义为枚举形式
 * 错误码列表:
 *  10: 通用
 *      001:参数格式校验
 *  11: 商品
 *  12: 订单
 *  13: 购物车
 *  14: 物流
 *
 *
 */
public enum BizCodeEnume {
    UNKNOW_EXCEPTION(10000,"系统未知异常"),
    VAILD_EXCEPTION(10001,"参数格式校验失败");

    private int code;
    private String msg;
    BizCodeEnume(int code, String msg){
        this.code = code;
        this.msg = msg;
    }

    public int getCode() {
        return code;
    }

    public String getMsg() {
        return msg;
    }
}

参考结构
在这里插入图片描述

建一个全局异常捕获类

这样我们在com.example.jsr303.controller下的异常都能被这个类捕获到进行统一处理
在这里插入图片描述

package com.atguigu.gulimall.product.exception;

import com.atguigu.common.exception.BizCodeEnume;
import com.atguigu.common.utils.R;
import lombok.extern.slf4j.Slf4j;
import org.springframework.validation.BindingResult;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;

import java.util.HashMap;
import java.util.Map;

/**
* 集中处理所有异常
*/
@Slf4j
//@ResponseBody
//@ControllerAdvice(basePackages = "com.atguigu.gulimall.product.controller")
@RestControllerAdvice(basePackages = "com.atguigu.gulimall.product.controller")
public class GulimallExceptionControllerAdvice {


   @ExceptionHandler(value= MethodArgumentNotValidException.class)
   public R handleVaildException(MethodArgumentNotValidException e){
       log.error("数据校验出现问题{},异常类型:{}",e.getMessage(),e.getClass());
       BindingResult bindingResult = e.getBindingResult();

       Map<String,String> errorMap = new HashMap<>();
       bindingResult.getFieldErrors().forEach((fieldError)->{
           errorMap.put(fieldError.getField(),fieldError.getDefaultMessage());
       });
       return R.error(BizCodeEnume.VAILD_EXCEPTION.getCode(),BizCodeEnume.VAILD_EXCEPTION.getMsg()).put("data",errorMap);
   }

   @ExceptionHandler(value = Throwable.class)
   public R handleException(Throwable throwable){

       log.error("错误:",throwable);
       return R.error(BizCodeEnume.UNKNOW_EXCEPTION.getCode(),BizCodeEnume.UNKNOW_EXCEPTION.getMsg());
   }
}

controller层这么编写就可以
因为因为验证失败抛出的MethodArgumentNotValidException会被我们的异常捕获器捕获处理

   @PostMapping("/save")
   public String save(@Valid @RequestBody TestEntity testEntity){
           testService.save(testEntity);
           return "ok";
   }

在这里插入图片描述

3 JSR303分组校验

思路:
分组校验(多场景的复杂校验)
1)、 @NotBlank(message = “品牌名必须提交”,groups = {AddGroup.class,UpdateGroup.class})
给校验注解标注什么情况需要进行校验
2)、@Validated({AddGroup.class})
3)、默认没有指定分组的校验注解 在分组校验情况@Validated({AddGroup.class})下不生效,只会在@Validated生效;

1 在 common包下创建一个vaild包

里面定义AddGroup接口和UpdateGroup接口
在这里插入图片描述

2 在实体类上定义组分类

形式(groups = {组接口.class})

@Data
@TableName("test")
public class TestEntity implements Serializable {

    //忽略掉 设置序列化和反序列化的serialVersionUID
    //想了解一下可以去这篇博客 https://blog.csdn.net/qq_45503106/article/details/107950914
    private static final long serialVersionUID = 1L;

    /**
     * test表主键ID
     */
    @TableId(type = IdType.AUTO)
    private Integer id;
    /**
     *  姓名
     */
    @NotEmpty(message = "姓名不能为空",groups = {AddGroup.class})
    private String name;

    /**
     * 邮箱
     */
    @NotEmpty(message = "邮箱不能为空",groups = {UpdateGroup.class})
    private String email;

}

3 在controller层上使用@vaildated(接口.class)开启分组校验(这时候如果实体类上没有指定组的校验将会失效)

    @PostMapping("/save")
    public String save(@Validated(AddGroup.class) @RequestBody TestEntity testEntity){
            testService.save(testEntity);
            return "ok";
    }

在这里插入图片描述

4 JSR303 自定义校验注解

思路:

  • 5)、自定义校验
  •  1)、编写一个自定义的校验注解
    
  •  2)、编写一个自定义的校验器 ConstraintValidator
    
  •  3)、关联自定义的校验器和自定义的校验注解
      *      @Documented
      * @Constraint(validatedBy = { ListValueConstraintValidator.class【可以指定多个不同的校验器,适配不同类型的校验】 })
      * @Target({ METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER, TYPE_USE })
      * @Retention(RUNTIME)
      * public @interface ListValue {
    

1 在自己对应实体类上编写自定义注解

在这里插入图片描述

    /**
     * 邮箱
     */

    @ListValue(vals={"1","2"},groups = {AddGroup.class})
    private String email;

2 编写注解接口

参考结构
在这里插入图片描述

自定义注解编写ListValue

@Documented
//指定校验器 这里指定我们自定义的ListValueConstraintValidator
@Constraint(validatedBy = { ListValueConstraintValidator.class })
@Target({ METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER, TYPE_USE })
@Retention(RUNTIME)
public @interface ListValue {
    //使用我们自己定义的错误信息 在common模块下的resources中
//    String message() default "{com.atguigu.common.valid.ListValue.message}";
    String message() default "值必须是0或者1";

    Class<?>[] groups() default { };

    Class<? extends Payload>[] payload() default { };

    //预先准备的值 vals={0,1}
    String[] vals() default {};
}

//对应实体类的注解
//@ListValue(vals={0,1},groups = {AddGroup.class, UpdateStatusGroup.class})

3 编写注解校验器ListValueConstraintValidator


//对应实体类的注解
//@ListValue(vals={0,1},groups = {AddGroup.class, UpdateStatusGroup.class})
public class ListValueConstraintValidator implements ConstraintValidator<ListValue,String> {

    private Set<String> set = new HashSet<>();
    //初始化方法
    @Override
    public void initialize(ListValue constraintAnnotation) {

        //从注解上获取值 遍历将其存入一个set
        String[] vals = constraintAnnotation.vals();
        for (String val : vals) {
            set.add(val);
        }

    }
    //判断是否校验成功
    /**
     *
     * @param value 需要校验的值
     * @param context
     * @return
     */
    @Override
    public boolean isValid(String value, ConstraintValidatorContext context) {

        //拿到传入的值 判断set中是否包含这个值
        //不包含返回false
        return set.contains(value);
    }
}


4 controller层开启

    @PostMapping("/save")
    public String save(@Validated({AddGroup.class, UpdateGroup.class}) @RequestBody TestEntity testEntity){
            testService.save(testEntity);
            return "ok";
    }

在这里插入图片描述

总结

controller开启Vailid或者Validated进行校验 -> 到实体类的校验注解查看校验策略(分组 和注解内容,如notnull等) ->分支1 非定义注解 验证后通过或是返回错误信息——>分支2 自定义注解 去自定义注解中找到校验器
然后进行校验判断

  • 3
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

qwecxzz

鸡腿

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值