文章目录

  • 约束注解(大多为 `javax.validation.constraints` 下的注解)
  • 启用校验
  • 使用
  • 消息
  • 示例
  • 进阶
  • 总结

后端数据校验(JSR303/JSR-349、javax validation、hibernate validation、spring validation)

后端数据校验如:请求参数不能为null、数值至少为5、email参数符合邮箱地址规则等,通常涉及到上述几种工具,其区别:

  • JSR303/JSR-349、javax validation:JSR303是一项标准、JSR-349是其升级版本,添加了一些新特性,他们规定一些校验规范即校验注解,如@Null,@NotNull,@Pattern,他们位于javax.validation.constraints包下,只提供规范不提供实现
  • hibernate validation是对该规范的实践(不要将hibernate和数据库orm框架联系在一起),他提供了相应的实现,并增加了一些其他校验注解,如@Email,@Length,@Range等等,他们位于org.hibernate.validator.constraints包下
  • spring对hibernate validation进行了二次封装以方便使用,其在springmvc模块中添加了自动校验并将校验信息封装进了特定的类中、还支持指定groups以进行分组校验。以此可见,下面说到的分组是Spring validation提供的功能。
    在使用时最好用规范,这样可以在不改变代码的情况下选用其他具体实现。

maven依赖:

        <dependency><groupId>javax.validation</groupId><artifactId>validation-api</artifactId></dependency><!-- https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-validation -->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-validation</artifactId><version>2.2.5.RELEASE</version>
</dependency>

约束注解(大多为 javax.validation.constraints 下的注解)

javax Validation v2.0规范中已经定义了22个常用的标记。包括:@NotNull、@NotBlank、@Email、@Pattern、@AssertTrue、@Min、@DecimalMin、@Negative、@Size、@Past、@Future等

启用校验

通常是针对Controller Handler(即Controller中接收http request的方法)启用参数校验,当然也可以对自己创建的对象启用参数校验

启用自动校验可以通过@Validated( org.springframework.validation.annotation.Validated )或@Valid( javax.validation.constraints.Valid )完成

也可通过 Validation.buildDefaultValidatorFactory().getValidator() 获取一个 Validator 来手动启用校验

使用

1、指定约束:在Bean中的字段上加上@NotNull等约束注解(若方法参数是一个Bean对象如courseEntityDTO,注解可以放在字段声明上,也可以放在字段的get方法上),或直接对方法参数加上约束注解(如 public void test(@NotNull String courseId) {...} );

2、启用约束:

若是对请求体的值(@RequestBody 参数)做验证,则在Controller请求方法的Bean参数前加上@Validated即可(也可用@Valid,但多用@Validated)。示例:

@PostMapping("/courses")
public ApiBaseResp<Boolean> importCourses(@Validated @RequestBody ImportCoursesDto importCoursesDto);

若是对非请求体值(如@RequestParam 参数)的验证,则将@Validated放在方法所在的类上,示例:

@Validated
@Data
@Configuration
@ConfigurationProperties(prefix = "sensestudy.security.jwt")
public class JwtSettings {@NotBlankprivate String tokenIssuer;@NotBlankprivate String tokenSigningKey;@NotNullprivate Integer tokenExpirationTimeMinutes;@NotNullprivate Integer refreshTokenExpireTimeMinutes;/** 若 */@NotBlankprivate String domainForCookie = "*";@Validprivate Cookie cookie=new Cookie();}
@Data
class Cookie {@NotBlankprivate String domain;
}

消息

message参数用于指定出错时的提示信息。如果是{xxxx}的格式,则可用于本地化,如果找不到,就作为一般性描述。如果没有大括号,就是一般性的描述。

缺省的内置的限制标记都有相应的{xxxx}说明,例如 {javax.validation.constraints.NotNull.message}。

未指定则会用注解的默认值,如{javax.validation.constraints.NotNull.message};若自己指定,则值(可以用EL表达式):

既可以是字面值,如"name is not present"

也可以包含EL表达式,如 @Size(min=0, max=5, message="value ${validatedValue} should be between [{min},{max}]") ,message中可以引用变量:

对于注解自身有的属性的值可以通过 {属性名} 引用
引用所传的值用 ${validatedValue}

若要进行消息国际化显示,则可通过LocalValidatorFactoryBean指定消息配置源,如:

@Configuration
public class ValidationConfig{@Value(value = "${spring.messages.basename}")private String basename;@Bean(name = "messageSource")public ResourceBundleMessageSource messageSource() {ResourceBundleMessageSource messageSource = new ResourceBundleMessageSource();String[] basenames=basename.split(",");messageSource.setBasenames(basenames);messageSource.setDefaultEncoding("UTF-8");return messageSource;}@Beanpublic LocalValidatorFactoryBean getValidator() {LocalValidatorFactoryBean bean = new LocalValidatorFactoryBean();bean.setValidationMessageSource(messageSource());return bean;}
}

示例

@AssertTrue:用于修饰方法或属性,要求方法名或属性名必须以 is 开头

//用于修饰方法或属性,要求方法名或属性名必须以 is 开头//用于方法
@AssertTrue(message = "the value of grant type parameter must be 'clientcredentials'")//校验失败时默认报'grantType只能为true',可以通过message指定自定义消息
private boolean isGrantTypeRight() {return getGrant_type().equals(GrantTypeEnum.clientcredentials);}//用于属性
@AssertTrue
private boolean isCourseDisabled;

进阶

  • 嵌套的内部对象的验证:在Person p有个字段List cars,若在验证p时同时要验证Car的price等字段,可以在cars前加上 @Valid 即可
  • 分组:在Bean中可以指定字段验证所属的groups、在请求参数中可以指定应用哪种groups进行验证,只会触发相应的groups进行验证;若未指定groups则默认属于组javax.validation.groups.Default,更多可参阅(分组)示例:
//Bean中的定义@Min(value = 18, groups = { Adult.class ,Default.class}) // groups限制触发此约束的条件,groups中无元素则默认为Default.classprivate Integer age;//Controller中的验证@PostMapping("/foo1")public String foo1(@RequestBody @Validated({ Adult.class }) Foo foo1) {System.out.println("------- res1 -------");return "foo1 done.";}
  • 分组顺序验证:通过@GroupSequence定义一个组SeqGroup,里面包含若干其他组,则SeqGroup可起顺序验证若干组的作用。示例:
public interface GroupA {}public interface GroupB {}@GroupSequence( { Default.class, GroupA.class, GroupB.class })
public interface SeqGroup {}
  • 可以不用在方法中与每个请求参数都对应一个BindingResult,而是拦截Controller异常进行处理:违背约束时会抛MethodArgumentNotValidException异常。示例:
@Slf4j
@ControllerAdvice
class GlobalControllerExceptionHandler {@ResponseBody@ExceptionHandler(Throwable.class)public String handleApiBindException(Throwable e) {String msg = e.getLocalizedMessage();log.error("param validation error", e);if (e instanceof BindException) {//            return handleApiBindException((BindException) e);}if (e instanceof MethodArgumentNotValidException) {BindingResult bindingResult2 = ((MethodArgumentNotValidException) e).getBindingResult();if (bindingResult2.hasErrors()) {for (FieldError fieldError : bindingResult2.getFieldErrors()) {System.out.println(String.format("%s %s %s %s", fieldError.getCode(), fieldError.getField(),fieldError.getDefaultMessage(), fieldError.getRejectedValue()));}}msg = ((MethodArgumentNotValidException) e).getBindingResult().getFieldErrors().stream().map(fe -> String.format("'%s'%s", fe.getField(), fe.getDefaultMessage())).collect(Collectors.joining(","));}if (e instanceof ConstraintViolationException) {//            return handleApiConstraintViolationException((ConstraintViolationException) e);}return msg;}
}

手动启用验证:通过javax.validation.ValidatorFactory获取一个Validator然后进行验证。此法支持分组但对部分约束(如@Email)不生效。该Validator可针对整个类或指定部分字段进行验证,还支持指定分组等。

Validator validator = Validation.buildDefaultValidatorFactory().getValidator();
validator.validate(new Foo(), Default.class).forEach(e -> {System.err.println(e.getPropertyPath() + " " + e.getMessage());//+ " " + e.getInvalidValue()
});
  • 自定义注解:

    • 定义自定义注解,该注解须被@Constraint修饰以指定该自定义注解对应的注解处理器
package com.marchon.learning.validation.custom_annotation;import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;import javax.validation.Constraint;
import javax.validation.Payload;@Target({ ElementType.METHOD, ElementType.FIELD, ElementType.ANNOTATION_TYPE })
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = CheckCaseValidator.class)
@Documented
public @interface CheckCase {// @Constraint要求必须有以下三个方法String message() default "'${validatedValue}' not {caseMode} case";// "com.marchon.learning.validation.constraintts.checkcase";Class<?>[] groups() default {};Class<? extends Payload>[] payload() default {};// 以下方法为其他自定义方法CaseMode caseMode();public enum CaseMode {UPPER, LOWER}
}
  • 定义自定义注解的处理器,该处理器须实现ConstraintValidator接口
package com.marchon.learning.validation.custom_annotation;import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;import com.marchon.learning.validation.custom_annotation.CheckCase.CaseMode;public class CheckCaseValidator implements ConstraintValidator<CheckCase, String> {//两参数分别为注解类型、注解作用目标的属性类型private CaseMode caseMode;@Overridepublic void initialize(CheckCase constraintAnnotation) {this.caseMode = constraintAnnotation.caseMode();}@Overridepublic boolean isValid(String value, ConstraintValidatorContext constraintContext) {if (null == value)return true;if (caseMode == CaseMode.UPPER)return value.equals(value.toUpperCase());elsereturn value.equals(value.toLowerCase());}}

使用示例

    @CheckCase(caseMode = CaseMode.UPPER)@NotBlankprivate String name;

总结

可见用的最多的是spring的 @Validated 、其次是javax的 @Valid ,其多数基本功能相似,但在注解适用位置、分组、嵌套验证支持上有区别。

注解适用位置:前者可用于修饰类型、方法、方法参数上,不能修饰成员变量;后者可用于方法、方法参数、成员变量、构造函数上。能否用于成员属性决定了是否支持嵌套验证。

分组:@Validated 支持,@Valid 不支持。

嵌套验证:@Validated 不支持,@Valid 支持。

更多详情参阅:

https://www.cnkirito.moe/spring-validation/

https://blog.csdn.net/qq_27680317/article/details/79970590

https://blog.csdn.net/flowingflying/article/details/78150015

Spring Boot学习笔记(二十一)Spring boot 数据校验 @Validated、@Valid用法详解相关推荐

  1. python的messagebox的用法_Python GUI编程学习笔记之tkinter中messagebox、filedialog控件用法详解...

    本文实例讲述了Python GUI编程学习笔记之tkinter中messagebox.filedialog控件用法.分享给大家供大家参考,具体如下: 相关内容: messagebox 介绍 使用 fi ...

  2. python3.4学习笔记(二十一) python实现指定字符串补全空格、前面填充0的方法

    python3.4学习笔记(二十一) python实现指定字符串补全空格.前面填充0的方法 Python zfill()方法返回指定长度的字符串,原字符串右对齐,前面填充0. zfill()方法语法: ...

  3. JDBC学习笔记01【JDBC快速入门、JDBC各个类详解、JDBC之CRUD练习】

    黑马程序员-JDBC文档(腾讯微云)JDBC笔记.pdf:https://share.weiyun.com/Kxy7LmRm JDBC学习笔记01[JDBC快速入门.JDBC各个类详解.JDBC之CR ...

  4. Mr.J-- jQuery学习笔记(二十一)--模拟微博页面

    先看之前的节点操作方法:Mr.J-- jQuery学习笔记(二十)--节点操作方法 Mr.J-- jQuery学习笔记(五)--属性及属性节点 Mr.J-- jQuery学习笔记(十一)--事件委托  ...

  5. kvm虚拟化学习笔记(二十一)之KVM性能优化学习笔记

    本学习笔记系列都是采用CentOS6.x操作系统,KVM虚拟机的管理也是采用virsh方式,网上的很多的文章都基于ubuntu高版本内核下,KVM的一些新的特性支持更好,本文只是记录了CentOS6. ...

  6. ESL3.5 学习笔记(主成分回归,偏最小二乘回归步骤详解)

    3.5 运用派生输入方向的方法 这是一篇有关<统计学习基础>,原书名The Elements of Statistical Learning的学习笔记,该书学习难度较高,有很棒的学者将其翻 ...

  7. Spring框架学习笔记05:Spring AOP基础

    文章目录 一.Spring AOP (一)AOP基本含义 (二)AOP基本作用 (三)AOP与OOP (四)AOP使用方式 (五)AOP基本概念 任务:骑士执行任务前和执行任务后,游吟诗人唱赞歌 (一 ...

  8. Spring框架学习笔记06:Spring JdbcTemplate入门

    文章目录 一.JdbcTemplate概述 (一)JdbcTemplate作用 (二)JdbcTemplate操作数据库常用方法 二.数据库连接池 (一)连接池原理 (二)常见数据库连接池 1.DBC ...

  9. linux驱动开发学习笔记二十一:异步通知

    一.异步通知简介 我们首先来回顾一下"中断",中断是处理器提供的一种异步机制,我们配置好中断以后就可以让处理器去处理其他的事情了,当中断发生以后会触发我们事先设置好的中断服务函数, ...

  10. opencv学习笔记二十一:使用HSV颜色空间实现颜色识别

    一.颜色空间介绍        RGB 颜色空间是大家最熟悉的颜色空间,即三基色空间,任何一种颜色都可以由该三种 颜色混合而成.然而一般对颜色空间的图像进行有效处理都是在 HSV 空间进行的,HSV( ...

最新文章

  1. python中词云图是用来描述_python中实现词云图
  2. android开源库收集
  3. 如异界守塔的伪原创工具
  4. 初三中考比一模能提高多少分?
  5. 网站源代码遭公开,2.43亿名巴西公民个人数据遭泄露
  6. JQ简单二级导航,加子导航栏
  7. 委托、事件--委托介绍篇
  8. Android Studio 工程项目的结构
  9. ubuntu播放文件需要MPEG-4 AAC解码器
  10. python的源代码下载_官方下载python源码,编译linux版本的python
  11. 国内优秀开源镜像站汇总
  12. python 调用函数
  13. linux下数学公式的编辑器,如何利用开源中国Markdown/编辑器优雅的写出数学公式?(KaTeX公式使用篇)...
  14. FMI飞马网【线上直播】持续集成在京东研发的落地及应用
  15. Python中print的用法
  16. 如何解决在线医疗语音问诊的技术难题 | 以眩晕问诊为例
  17. java制作局域网聊天器_基于Java的局域网聊天工具
  18. bootloader 和 启动模式的一些理解
  19. 怎样看 计算机类b0809,计算机类(B0809、A0812、A0854)为什么是神?
  20. 第1章 数字图像处理绪论

热门文章

  1. 小卡机器人积木教程_这个会动的智能积木机器人分分钟让孩子爱不释手!秒杀家中99%的益智玩具!| 团...
  2. “圆球池塘”——动态交互作品
  3. better-scroll的使用以及常见的api总结
  4. 单片机微型计算机原理及接口技术,单片机微型计算机原理及接口技术课后习题答案第4章.docx...
  5. 第三代航空轴承钢产品 css-42l,航空轴承钢的发展及热处理技术(一)
  6. 多宽带联网(五) 仅使用Windows进行多宽带叠加
  7. 8位bmp文件获取像素
  8. 中国人民大学计算机考研分数线2020,中国人民大学2020考研国家线-复试分数线-录取分数线 - 希赛网...
  9. Microsoft Edge浏览器黑色背景修改
  10. JScript服务器运行,jscript模拟的“控制台”程序Web服务器教程