使用spring-boot-starter-validation实现入参校验

使用 spring-boot-starter-validation实现入参校验

引入依赖

复制代码
<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-validation</artifactId>
</dependency>

定义分组

由于新增、更新、查询、删除都可以使用一个实体接受入参,而不同情况需要校验的情况也不同,比如,新增时不需要校验id必填,删除时需要校验id必填,分组可以理解成不同策略,不同分组下可以使用不同的校验规则

Java 复制代码
package com.param_validate_demo.groups;

import javax.validation.groups.Default;

public interface Insert  extends Default {
}

定义校验规则

Java 复制代码
package com.param_validate_demo.model;

import com.param_validate_demo.annotation.AgeValidate;
import com.param_validate_demo.annotation.EnumValidate;
import com.param_validate_demo.enums.GenderEnum;
import com.param_validate_demo.enums.RoleEnum;
import com.param_validate_demo.groups.Insert;
import com.param_validate_demo.groups.Query;
import com.param_validate_demo.groups.Update;
import lombok.Data;

import javax.validation.constraints.NotNull;
import java.math.BigInteger;
import java.util.List;

@Data
@AgeValidate(groups = {Insert.class, Update.class})
public class User {
    private BigInteger id;
    @NotNull(groups = {Insert.class, Update.class, Query.class},message = "name 不能为空")
    private String name;
    private int age;
    private String address;
    private String phone;
    private String email;

    @EnumValidate(enumClass = GenderEnum.class, groups = {Insert.class, Update.class})
    private String gender;

    @EnumValidate(enumClass = RoleEnum.class, message = "入参roles中存在值不在枚举RoleEnum中", groups = {Insert.class, Update.class})
    private List<String> roles;
}

在 controller 中使用

如下,新增使用 Insert.class 规则,查询使用 Query.class 规则

Java 复制代码
package com.param_validate_demo.controller;


import com.param_validate_demo.groups.Insert;
import com.param_validate_demo.groups.Query;
import com.param_validate_demo.model.User;
import org.springframework.stereotype.Controller;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

@Controller
@RequestMapping("/user")
public class UserController {

    @PostMapping("/add")
    @ResponseBody
    public String addUser(@Validated(Insert.class) @RequestBody User user) {
        return "新增成功";
    }

    @PostMapping("/query")
    @ResponseBody
    public String query(@Validated(Query.class) @RequestBody User user) {
        return "查询成功";
    }
}

自定义校验

有些时候框架提供的校验不满足当前业务,需要自己定义校验规则,比方上述user中的roles可以传入一些角色,但是这个角色必须在系统定义的枚举中,如下自定义一个校验枚举的校验器

① 新增注解
Java 复制代码
package com.param_validate_demo.annotation;

import com.param_validate_demo.validator.EnumValidator;

import javax.validation.Constraint;
import javax.validation.Payload;
import java.lang.annotation.*;

@Documented
@Constraint(validatedBy = EnumValidator.class)
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD})
public @interface EnumValidate {

    /**
     * 是否必填,默认必填
     */
    boolean required() default true;

    Class<? extends Enum<?>> enumClass();

    /**
     * 必要参数,否则报错
     */
    String message() default "";

    /**
     * 必要参数,否则报错
     * javax.validation.ConstraintDefinitionException: HV000074: com.param_validate_demo.annotation.EnumValidate contains Constraint annotation, but does not contain a groups parameter.
     */
    Class<?>[] groups() default { };

    /**
     * 必要参数,否则报错
     */
    Class<? extends Payload>[] payload() default {};

}
② 新增校验器
Java 复制代码
package com.param_validate_demo.validator;

import com.param_validate_demo.annotation.EnumValidate;
import com.param_validate_demo.enums.MyEnum;

import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Objects;
import java.util.stream.Collectors;

public class EnumValidator implements ConstraintValidator<EnumValidate, Object> {

    private boolean required;
    private Class<? extends Enum<?>> enumClass;

    @Override
    public void initialize(EnumValidate constraintAnnotation) {
        this.required = constraintAnnotation.required();
        this.enumClass = constraintAnnotation.enumClass();
    }

    @Override
    public boolean isValid(Object param, ConstraintValidatorContext constraintValidatorContext) {
        // 如果不需要校验,并且参数为空则不校验,直接放行,否则如果 required = false、param不是空也要校验
        if (!required && Objects.isNull(param)) {
            return true;
        }
        if (Objects.isNull(enumClass) || Objects.isNull(param)) {
            return false;
        }
        if (MyEnum.class.isAssignableFrom(enumClass)) {
            MyEnum[] myEnums = (MyEnum[]) enumClass.getEnumConstants();
            if (Objects.isNull(myEnums)) {
                return false;
            }
            List<String> codes = Arrays.stream(myEnums).map((MyEnum::getCode)).collect(Collectors.toList());
            if (param instanceof Collection) {
                Collection<?> collection = ((Collection<?>) param);
                return collection.stream().allMatch((Object o) -> codes.contains(o.toString()));
            } else if (param.getClass().isArray()) {
                return Arrays.stream((Object[]) param).allMatch((Object o) -> codes.contains(o.toString()));
            } else {
                // 单个参数
                return codes.stream().anyMatch((String item) -> Objects.equals(item, param));
            }
        }
        return false;
    }
}
枚举实现的接口如下
Java 复制代码
package com.param_validate_demo.enums;

public interface MyEnum {
    String getCode();
    String getDesc();
}
角色枚举
Java 复制代码
package com.param_validate_demo.enums;

public enum RoleEnum implements MyEnum {
    USER_MANAGE("user_manage", "用户管理员"), ORDER_MANAGE("order_manage", "订单管理员");
    private String code;
    private String desc;

    RoleEnum(String code, String desc) {
        this.code = code;
        this.desc = desc;
    }

    @Override
    public String getCode() {
        return this.code;
    }

    @Override
    public String getDesc() {
        return this.desc;
    }
}
性别枚举
Java 复制代码
package com.param_validate_demo.enums;

public enum GenderEnum implements MyEnum{
    MALE("male","男"),FEMALE("female","女");
    private String code;
    private String desc;

    GenderEnum(String code, String desc) {
        this.code = code;
        this.desc = desc;
    }

    @Override
    public String getCode() {
        return this.code;
    }

    @Override
    public String getDesc() {
        return this.desc;
    }
}
③ 使用注解
Java 复制代码
@EnumValidate(enumClass = GenderEnum.class, groups = {Insert.class, Update.class})
private String gender;

@EnumValidate(enumClass = RoleEnum.class, message = "入参roles中存在值不在枚举RoleEnum中", groups = {Insert.class, Update.class})
private List<String> roles;

参考:https://developer.aliyun.com/article/1670620

上述源代码:https://gitee.com/husong_zone/param_validate_demo

相关推荐
seabirdssss21 小时前
使用Spring Boot DevTools快速重启功能
java·spring boot·后端
喂完待续21 小时前
【序列晋升】29 Spring Cloud Task 微服务架构下的轻量级任务调度框架
java·spring·spring cloud·云原生·架构·big data·序列晋升
benben04421 小时前
ReAct模式解读
java·ai
轮到我狗叫了1 天前
牛客.小红的子串牛客.kotori和抽卡牛客.循环汉诺塔牛客.ruby和薯条
java·开发语言·算法
yudiandian20141 天前
【QT 5.12.12 下载 Windows 版本】
开发语言·qt
高山有多高1 天前
详解文件操作
c语言·开发语言·数据库·c++·算法
狂奔的sherry1 天前
单例模式(巨通俗易懂)普通单例,懒汉单例的实现和区别,依赖注入......
开发语言·c++·单例模式
Volunteer Technology1 天前
三高项目-缓存设计
java·spring·缓存·高并发·高可用·高数据量
EnigmaCoder1 天前
【C++】引用的本质与高效应用
开发语言·c++
栗子~~1 天前
bat脚本- 将jar 包批量安装到 Maven 本地仓库
java·maven·jar