element form自定义校验_SpringBoot分组校验及自定义校验注解

b744df83096341a801fb8daae7909741.png

前言

  在日常的开发中,参数校验是非常重要的一个环节,严格参数校验会减少很多出bug的概率,增加接口的安全性。在此之前写过一篇SpringBoot统一参数校验主要介绍了一些简单的校验方法。而这篇则是介绍一些进阶的校验方式。比如说:在某个接口编写的过程中肯定会遇到,当xxType值为A,paramA值必传。xxType值为B,paramB值必须传。对于这样的,通常的做法就是在controller加上各种if判断。显然这样的代码是不够优雅的,而分组校验及自定义参数校验,就是来解决这个问题的。

PathVariable参数校验

  Restful的接口,在现在来讲应该是比较常见的了,常用的地址栏的参数,我们都是这样校验的。

/**
 * 获取电话号码信息
 */
@GetMapping("/phoneInfo/{phone}")
public ResultVo phoneInfo(@PathVariable("phone") String phone){
    // 验证电话号码是否有效
    String pattern = "^[1][3,4,5,7,8][0-9]{9}$";
    boolean isValid =  Pattern.matches(pattern, phone);
    if(isValid){
        // 执行相应逻辑
        return ResultVoUtil.success(phone);
    } else {
        // 返回错误信息
        return ResultVoUtil.error("手机号码无效");
    }
}

很显然上面的代码不够优雅,所以我们可以在参数后面,添加对应的正则表达式phone:正则表达式来进行验证。这样就省去了在controller编写校验代码了。

/**
 * 获取电话号码信息
 */
@GetMapping("/phoneInfo/{phone:^[1][3,4,5,7,8][0-9]{9}$}")
public ResultVo phoneInfo(@PathVariable("phone") String phone){
    return ResultVoUtil.success(phone);
}

虽然这样处理后代码更精简了。但是如果传入的手机号码,不符合规则会直接返回404。而不是提示手机号码错误。错误信息如下:

90698439ad41533c255bdf20d98605f3.png

自定义校验注解

  我们以校验手机号码为例,虽然validation提供了@Pattern这个注解来使用正则表达式进行校验。如果被使用在多处,一旦正则表达式发生更改,则需要一个一个的进行修改。很显然为了避免做这样的无用功,自定义校验注解就是你的好帮手。

@Data
public class PhoneForm {

    /**
     * 电话号码
     */
    @Pattern(regexp = "^[1][3,4,5,7,8][0-9]{9}$" , message = "电话号码有误")
    private String phone;

}

  要实现一个自定义校验注解,主要是有两步。一是注解本身,二是校验逻辑实现类

PhoneVerify 校验注解

@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = PhoneValidator.class)
public @interface Phone {
 
    String message() default "手机号码格式有误";

    Class>[] groups() default {};

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

}

PhoneValidator 校验实现类

public class PhoneValidator implements ConstraintValidator<Phone, Object> {

    @Override
    public boolean isValid(Object telephone, ConstraintValidatorContext constraintValidatorContext) {
        String pattern = "^1[3|4|5|7|8]\\d{9}$";
        return Pattern.matches(pattern, telephone.toString());
    }
}

CustomForm 表单数据

@Data
public class CustomForm {

    /**
     * 电话号码
     */
    @Phone
    private String phone;

}

测试接口

@PostMapping("/customTest")
public ResultVo customTest(@RequestBody @Validated CustomForm form){
 return ResultVoUtil.success(form.getPhone());
}

注解的含义

@Target({ElementType.FIELD})

  注解是指定当前自定义注解可以使用在哪些地方,这里仅仅让他可以使用属性上。但还可以使用在更多的地方,比如说方法上、构造器上等等。

  • TYPE - 类,接口(包括注解类型)或枚举
  • FIELD - 字段(包括枚举常量)
  • METHOD - 方法
  • PARAMETER - 参数
  • CONSTRUCTOR - 构造函数
  • LOCAL_VARIABLE - 局部变量
  • ANNOTATION_TYPE -注解类型
  • PACKAGE - 包
  • TYPE_PARAMETER - 类型参数
  • TYPE_USE - 使用类型
@Retention(RetentionPolicy.RUNTIME)

  指定当前注解保留到运行时。保留策略有下面三种:

  • SOURCE - 注解只保留在源文件,当Java文件编译成class文件的时候,注解被遗弃。
  • CLASS - 注解被保留到class文件,但jvm加载class文件时候被遗弃,这是默认的生命周期。
  • RUNTIME - 注解不仅被保存到class文件中,jvm加载class文件之后,仍然存在。
@Constraint(validatedBy = PhoneValidator.class)

  指定了当前注解使用哪个校验类来进行校验。

分组校验

UserForm

@Data
public class UserForm {

    /**
     * id
     */
    @Null(message = "新增时id必须为空", groups = {Insert.class})
    @NotNull(message = "更新时id不能为空", groups = {Update.class})
    private String id;

    /**
     * 类型
     */
    @NotEmpty(message = "姓名不能为空" , groups = {Insert.class})
    private String name;

    /**
     * 年龄
     */
    @NotEmpty(message = "年龄不能为空" , groups = {Insert.class})
    private String age;
    
}

Insert分组

public interface Insert {
}

Update分组

public interface Update {
}

测试接口

/**
 * 添加用户
 */
@PostMapping("/addUser")
public ResultVo addUser(@RequestBody @Validated({Insert.class}) UserForm form){
   // 选择对应的分组进行校验
    return ResultVoUtil.success(form);
}

/**
 * 更新用户
 */
@PostMapping("/updateUser")
public ResultVo updateUser(@RequestBody @Validated({Update.class}) UserForm form){
    // 选择对应的分组进行校验
    return ResultVoUtil.success(form);
}

测试结果

添加测试
644717abd9384b1b28c0097807fe2832.png
更新测试
cbcdd85a36dd6113cbd9fd41d29ffa19.png

顺序校验@GroupSequence

  在@GroupSequence内可以指定,分组校验的顺序。比如说@GroupSequence({Insert.class, Update.class, UserForm.class})先执行Insert校验,然后执行Update校验。如果Insert分组,校验失败了,则不会进行Update分组的校验。

@Data
@GroupSequence({Insert.class, Update.class, UserForm.class})
public class UserForm {

    /**
     * id
     */
    @Null(message = "新增时id必须为空", groups = {Insert.class})
    @NotNull(message = "更新时id不能为空", groups = {Update.class})
    private String id;

    /**
     * 类型
     */
    @NotEmpty(message = "姓名不能为空" , groups = {Insert.class})
    private String name;

    /**
     * 年龄
     */
    @NotEmpty(message = "年龄不能为空" , groups = {Insert.class})
    private String age;

}
测试接口
/**
* 编辑用户
*/
@PostMapping("/editUser")
public ResultVo editUser(@RequestBody @Validated UserForm form){
 return ResultVoUtil.success(form);
}
测试结果

  哈哈哈,测试结果其实是个死循环,不管你咋输入都会报错,小伙伴可以尝试一下哦。上面的例子只是个演示,在实际中还是别这样做了,需要根据具体逻辑进行校验。

自定义分组校验

  对于之前提到了当xxType值为A,paramA值必传。xxType值为B,paramB值必须传这样的场景。单独使用分组校验和分组序列是无法实现的。需要使用@GroupSequenceProvider才行。

自定义分组表单

@Data
@GroupSequenceProvider(value = CustomSequenceProvider.class)
public class CustomGroupForm {

    /**
     * 类型
     */
    @Pattern(regexp = "[A|B]" , message = "类型不必须为 A|B")
    private String type;

    /**
     * 参数A
     */
    @NotEmpty(message = "参数A不能为空" , groups = {WhenTypeIsA.class})
    private String paramA;

    /**
     * 参数B
     */
    @NotEmpty(message = "参数B不能为空", groups = {WhenTypeIsB.class})
    private String paramB;

    /**
     * 分组A
     */
    public interface WhenTypeIsA {

    }

    /**
     * 分组B
     */
    public interface WhenTypeIsB {

    }

}

CustomSequenceProvider

public class CustomSequenceProvider implements DefaultGroupSequenceProvider<CustomGroupForm> {

    @Override
    public List> getValidationGroups(CustomGroupForm form) {
        List> defaultGroupSequence = new ArrayList<>();
        defaultGroupSequence.add(CustomGroupForm.class);if (form != null && "A".equals(form.getType())) {
            defaultGroupSequence.add(CustomGroupForm.WhenTypeIsA.class);
        }if (form != null && "B".equals(form.getType())) {
            defaultGroupSequence.add(CustomGroupForm.WhenTypeIsB.class);
        }return defaultGroupSequence;
    }
}

测试接口

/**
 * 自定义分组
 */
@PostMapping("/customGroup")
public ResultVo customGroup(@RequestBody @Validated CustomGroupForm form){
    return ResultVoUtil.success(form);
}

测试结果

Type类型为A
6db70b5f26c2252295fe12bf9477790d.png
Type类型为B
71cefd0a960bf018ab8df031b2e753c7.png

小结一下

  GroupSequence注解是一个标准的Bean认证注解。正如之前,它能够让你静态的重新定义一个类的,默认校验组顺序。然而GroupSequenceProvider它能够让你动态的定义一个校验组的顺序。

注意的一个点

SpringBoot 2.3.x 移除了validation依赖需要手动引入依赖。

<dependency>
  <groupId>org.springframework.bootgroupId>
  <artifactId>spring-boot-starter-validationartifactId>
dependency>

总结

  个人的一些小经验,参数的非空判断,这个应该是校验的第一步了,除了非空校验,我们还需要做到下面这几点:

  • 普通参数 - 需要限定字段的长度。如果会将数据存入数据库,长度以数据库为准,反之根据业务确定。
  • 类型参数 - 最好使用正则对可能出现的类型做到严格校验。比如type的值是【0|1|2】这样的。
  • 列表(list)参数 - 不仅需要对list内的参数是否合格进行校验,还需要对list的size进行限制。比如说 100。
  • 日期,邮件,金额,URL这类参数都需要使用对于的正则进行校验。
  • 参数真实性 - 这个主要针对于 各种Id 比如说 userIdmerchantId,对于这样的参数,都需要进行真实性校验,判断系统内是有含有,并且对应的状态是否正常。

  参数校验越严格越好,严格的校验规则不仅能减少接口出错的概率,同时还能避免出现脏数据,从而来保证系统的安全性和稳定性。

错误的提醒信息需要友好一点哦,防止等下被前端大哥吐槽哦。

上期回顾

  • SpringBoot统一参数校验

结尾

  如果觉得对你有帮助,可以多多评论,多多点赞哦,也可以到我的主页看看,说不定有你喜欢的文章,也可以随手点个关注哦,谢谢。

  我是不一样的科技宅,每天进步一点点,体验不一样的生活。我们下期见!

cc77e2ed28da4d1d1141c4863ae652bd.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/422276.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

javascript 栈 Stack

栈是只允许在表尾进行插入&#xff0c;删除的线性表。特点后进先出。 下面将演示用数组实现的栈 栈初始化&#xff1a;创建一个空栈 Init:function(){this.STACKMAX 100;this.stack new Array(this.STACKMACK);this.top -1;return this.stack; } 判断栈空&#xff1a; 若栈为…

mysql 去除warning_zabbix监控mysql之去掉烦人的warning告警语句

使用zabbix自带模板对mysql进行监控时&#xff0c;发现mysql5.6以上版本在使用mysqladmin时会发出警告&#xff1a;“Warning: Using a password on the command line interface can be insecure.” 。这样zabbix服务端获取数值的时候&#xff0c;会带有该字符串&#xff0c;导…

前端学习(526):等分布局

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>等分布局</title><style>.col1{backgro…

mysql运营_为线上运营Mysql数据库设置从库

一、为mysql运营主库添加一个repl 账号[rootzabbix_server ~]# mysql -uroot -p -S /var/lib/mysql/mysql.sockEnter password:Welcome to the MySQL monitor. Commands end with ; or \g.Your MySQL connectionid is 15778982Server version:5.7.25MySQL Community Server (GP…

linux下svn客户端安装及环境配置

一、svn客户端安装及环境配置. 果所在的linux机器上没有安装svn客户端&#xff0c;则首先安装svn客户端&#xff1a; 1. subversion-1.4.3.tar.bz2 subversion-deps-1.4.3.tar.bz2 2. 使用 tar xvfj subversion-1.4.3.tar.bz2 tar xvfj subversion-deps-1.4.3.tar.bz2 解压这…

前端学习(527):等分布局第二种方案

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>等分布局</title><style>.parent{width…

c语言 feof_C语言 技能提升 系列文章(六)文件操作

C语言除了提供open()/ read()/ write()/ close() 这些基本的操作以外&#xff0c;还提供了下面几个非常有用的API。// 删除指定的文件int remove ( const char * filename ); // 重命名指定的文件int rename ( const char * oldname, const char * newname );// 以“wb”模式打…

mysql 字符转数值_深入MYSQL字符数字转换的详解

1、将字符的数字转成数字&#xff0c;比如’0’转成0可以直接用加法来实现例如&#xff1a;将pony表中的d 进行排序&#xff0c;可d的定义为varchar&#xff0c;可以这样解决select * from pony order by (d0)2、在进行ifnull处理时&#xff0c;比如 ifnull(a/b,’0′) 这样就会…

谷歌(Google)是怎样对待离世的Google员工的?

日期&#xff1a;2012-8-10 来源&#xff1a;GBin1.com 如果提起谷歌的福利来说&#xff0c;大家肯定首先想到免费的食品和理发&#xff0c;及其独立的医疗服务&#xff0c;或者超棒的食堂和大厨等等。但是你是否知道谷歌是怎么对待死亡的Google员工的呢&#xff1f; 和Google…

前端学习(529):等分布局存在间距得实现得解决方案

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>等分布局</title><style>.parent{heigh…

mysql utf8 4位_mysql中utf8和utf8mb4区别

MySQL在5.5.3之后增加了这个utf8mb4的编码&#xff0c;mb4就是most bytes 4的意思&#xff0c;专门用来兼容四字节的unicode。好在utf8mb4是utf8的超集&#xff0c;除了将编码改为utf8mb4外不需要做其他转换。当然&#xff0c;为了节省空间&#xff0c;一般情况下使用utf8也就够…

easyUI 添加CheckBox选择到DataGrid

author YHC 这个教程向你展示如何放置一个checkbox 列到datagrid,这个复选框用户将可以选择 选中/取消选中 datagrid行数据. 查看 Demo 添加一个checkbox 列我们仅仅需要添加一个列的checkbox 属性设置它为true,代码看上去就像这些: <table id"tt" title"Che…

前端学习(530):等分布局得间距方案第二种方式

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>等分布局</title><style>.parent{width…

mysql error报错_MySQL报错:Error writing file '/tmp/MY4WYVlC' (Errcode: 28 - No space left on device)...

MySQL报错&#xff1a;Error writing file /tmp/MY4WYVlC (Errcode: 28 - No space left on device)一、问题描述1、MySQL好好的&#xff0c;突然就不行了&#xff0c;信息如下&#xff1a;系统内部错误.错误信息&#xff1a;org.springframework.jdbc.UncategorizedSQLExcepti…

xcode中工程引用设置

在A工程中引用B工程&#xff0c;需对A工程进行如下三点设置&#xff1a; 1、Build Settings-->Header Search Paths,此处采用相对路径&#xff0c;功能为除A工程外其它头文件在编译时的搜索路径; 2、Build Phases-->Target Dependencies.暂不知道将B.a加到此处的作用是什…

mysql函数 用来查询匹配不到的数据_erlang连接mysql数据库后为什么fetch说匹配不到这个函数...

展开全部需要注意&#xff0c;官方放在googlecode的测试代码已经旧e69da5e887aa62616964757a686964616f31333337383865了&#xff0c;fetch接口实际上需要一个二进制list&#xff0c;所以是[<>]格式&#xff0c;而不是<>格式。以下是增、删、改、查操作的测试代码&…

添加组合索引时,做相等运算字段应该放在最前面

有一个通常的误解&#xff0c;觉得应该把选择性高的字段放在最前面&#xff0c;这通常只是针对一个字段的索引&#xff0c;对于组合索引&#xff0c;常常要把做等式运算的字段放在最前面&#xff0c;看看测试 USE AdventureWorks GOCREATE TABLE demo1 ( id INT identity(1,1)…