后端数据校验(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>

约束注解:@NotNull、@NotBlank、@Email、@Pattern等(大多为 javax.validation.constraints 下的注解)

启用校验:通常是针对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 { @NotBlank
private String tokenIssuer; @NotBlank
private String tokenSigningKey; @NotNull
private Integer tokenExpirationTimeMinutes; @NotNull
private Integer refreshTokenExpireTimeMinutes; /** 若 */
@NotBlank
private String domainForCookie = "*"; @Valid
private Cookie cookie=new Cookie(); }
@Data
class Cookie {
@NotBlank
private String domain;
}

3、其他

1、message中引用内建变量的值(EL表达):可以通过注解的message指定不符合约束时返回的字符串信息,如 @Size(min=, max=, message="value ${validatedValue} should be between [{min},{max}]") ,message中可以引用变量:

对于注解自身有的属性可以通过 {属性名} 引用

引用所传的值用 ${validatedValue}

进阶

  • 嵌套的内部对象的验证:在Person p有个字段List<Car> 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.class
    private 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
      }
      }

      CheckCase

    • 定义自定义注解的处理器,该处理器须实现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; @Override
      public void initialize(CheckCase constraintAnnotation) {
      this.caseMode = constraintAnnotation.caseMode();
      } @Override
      public boolean isValid(String value, ConstraintValidatorContext constraintContext) { if (null == value)
      return true; if (caseMode == CaseMode.UPPER)
      return value.equals(value.toUpperCase());
      else
      return value.equals(value.toLowerCase());
      } }

      CheckCase Validator

    • 使用示例
          @CheckCase(caseMode = CaseMode.UPPER)
      @NotBlank
      private String name;

总结

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

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

分组:前者支持,后者不支持。

嵌套验证:前者不支持,后者支持。

更多详情参阅:https://www.cnkirito.moe/spring-validation/https://blog.csdn.net/qq_27680317/article/details/79970590

Java web小记的更多相关文章

  1. 高效 Java Web 开发框架 JessMA v3.5.1

    JessMA 是功能完备的高性能 Full-Stack Web 应用开发框架,内置可扩展的 MVC Web 基础架构和 DAO 数据库访问组件(内部已提供了 Hibernate.MyBatis 与 J ...

  2. 高效 Java Web 开发框架 JessMA v3.4.1

    JessMA 是功能完备的高性能 Full-Stack Web 应用开发框架,内置可扩展的 MVC Web 基础架构和 DAO 数据库访问组件(内部已提供了 Hibernate.MyBatis 与 J ...

  3. java web 之客户关系管理系统

    这个周末真的是觉得自己学会了一个比较高大上的本领,为什么这么觉得呢?那是因为星期六的时候觉得自己可以看看源码能做出来,可是让我头疼的是花费了一上午的时间还是没有弄出来,还好上天给了我机会,要是没有老师 ...

  4. Java Web中的中文编码

    Java Web开发中经常会遇到中文编码问题,那么为什么需要编码呢?因为人类需要表示的符号太多,无法用1个字节来表示,而计算机中存储信息最小单元为1个字节.所以必须指定char与byte之间的编码规则 ...

  5. java web后台开发SSM框架(Spring+SpringMVC+MyBaitis)搭建与优化

    一.ssm框架搭建 1.1创建项目 新建项目后规划好各层的包. 1.2导入包 搭建SSM框架所需包百度云链接:http://pan.baidu.com/s/1cvKjL0 1.3整合spring与my ...

  6. JAVA WEB项目中各种路径的获取

    JAVA WEB项目中各种路径的获取 标签: java webpath文件路径 2014-02-14 15:04 1746人阅读 评论(0) 收藏 举报  分类: JAVA开发(41)  1.可以在s ...

  7. JAVA WEB WITH IDEA

    本文主要介绍使用IDEA开发环境,创建JAVA WEB 工程,并介绍war包的制作过程. 1 创建MAVEN工程

  8. java WEB开发入门

    WEB开发入门 1 进入web JAVASE:标准- standard   JAVA桌面程序 GUI    SOCKET JAVAEE:企业-浏览器控制  web 2 软件结构 C/S :client ...

  9. Jenkins 2.16.3默认没有Launch agent via Java Web Start,如何配置使用

    问题:Jenkins 2.16.3默认没有Launch agent via Java Web Start,如下图所示,而这种启动方式在Windows上是最方便的. 如何设置才能让出来呢? 打开&quo ...

随机推荐

  1. Linux Linux程序练习五

    题目:编写两个进程a和b,利用共享内存技术,a向共享内存写字符串,b将从共享内存中读到的字符串在屏幕上打印出来. //创建共享内存区 #include <stdio.h> #include ...

  2. UTF-8 带签名和不带签名的区别

    就和字面上一样,带签名的UTF-8文件比不带签名的,在文件开头的地方就多了几个16进制字符--[EF BB BF ],这9个字符就是"签名",这样做的好处是让文本处理工具或者浏览器 ...

  3. eclipse使用

    Eclipse 是一个开放源代码的.基于 Java 的可扩展开发平台. Eclipse 是 Java 的集成开发环境(IDE),当然 Eclipse 也可以作为其他开发语言的集成开发环境,如C,C++ ...

  4. 当 IDENTITY_INSERT 设置为 OFF 时,不能为表中的标识列插入显式值

    {"当 IDENTITY_INSERT 设置为 OFF 时,不能向表 'OrderList' 中的标识列插入显式值"} 对于这个异常可以从两个角度来处理:A:数据库执行语句  B: ...

  5. MVC出错案例之一:主外键映射失败

    今天在编写DomainModel和DomainMapper,最后放到OnModelCreating中运行的时候,给我抛出了如下错误: One or more validation errors wer ...

  6. struct2cell

    函数功能:把结构体转换为元胞数组. 语法格式: c = struct2cell(s) 如果s是m*n(m行n列)的二维的结构体数组,每个结构体含有p个域,则转换得到一个p*m*n的元胞数组c. 如果s ...

  7. 对于JVM内存配置参数

    -Xmx:最大堆大小 -Xms:初始堆大小 -Xmn:年轻代大小 -XXSurvivorRatio:年轻代中Eden区与Survivor区的大小比值 年轻代5120m, Eden:Survivor=3 ...

  8. [CareerCup] 9.3 Magic Index 魔法序号

    9.3 A magic index in an array A[0.. .n-1] is defined to be an index such that A[i] = i. Given a sort ...

  9. 获取技能的成功经验和关于C语言学习的调查 2015528

    内容提要 你有什么技能比大多人(超过90%以上)更好?针对这个技能的获取你有什么成功的经验?与老师博客中的学习经验有什么共通之处? 有关C语言学习的调查 你是怎么学习C语言的?(作业,实验,教材,其他 ...

  10. AE 打开各种格式文件

    http://blog.sina.com.cn/s/blog_7a3fc90501016qrg.html /// <summary>/// 打开ShapeFile文件/// </su ...