Spring Boot (31) 数据验证
曾经参数的验证是这样的:
public String test(User user){
if(user == null){
throw new NullPointerException("user 不能为空");
}
if(user.getUserName() == null){
throw new NullPointerException("userName 不能为空");
}
if(user.getUserName().length() < 4 || user.getUserName().length()>10){
throw new RuntimeException("userName 长度小于4位或大于10位");
}
return "success";
}
随着参数的增加,格式的变化,校验数据有效性的代码愈发繁琐。
通过Spring boot来完成参数数据校验。
JSR-303注解介绍
这里只列举了javax.validation包下的注解,同理在spring-boot-starter-web包种也存在hibernate-validator验证包,里面包含了一些javax.validation没有的注解。
| 注解 | 说明 |
|---|---|
@NotNull |
限制必须不为null |
@NotEmpty |
验证注解的元素值不为 null 且不为空(字符串长度不为0、集合大小不为0) |
@NotBlank |
验证注解的元素值不为空(不为null、去除首位空格后长度为0),不同于@NotEmpty,@NotBlank只应用于字符串且在比较时会去除字符串的空格 |
@Pattern(value) |
限制必须符合指定的正则表达式 |
@Size(max,min) |
限制字符长度必须在 min 到 max 之间(也可以用在集合上) |
@Email |
验证注解的元素值是Email,也可以通过正则表达式和flag指定自定义的email格式 |
@Max(value) |
限制必须为一个不大于指定值的数字 |
@Min(value) |
限制必须为一个不小于指定值的数字 |
@DecimalMax(value) |
限制必须为一个不大于指定值的数字 |
@DecimalMin(value) |
限制必须为一个不小于指定值的数字 |
@Null |
限制只能为null(很少用) |
@AssertFalse |
限制必须为false (很少用) |
@AssertTrue |
限制必须为true (很少用) |
@Past |
限制必须是一个过去的日期 |
@Future |
限制必须是一个将来的日期 |
@Digits(integer,fraction) |
限制必须为一个小数,且整数部分的位数不能超过 integer,小数部分的位数不能超过 fraction (很少用) |
实体类
public class Book implements Serializable {
private Integer id;
@NotBlank(message = "name 不能为空")
@Length(min = 2, max = 10, message = "name 长度必须在{min}-{max}之间")
private String name;
@NotNull(message = "price 不能为空")
@DecimalMin(value = "0.1", message = "价格不能低于 {value}")
private BigDecimal price;
...
}
控制层
这些验证注解不仅仅可以放在controller上,也可以加在service层上。
//todo 开启数据有效性校验,添加在类上即为验证方法,添加在方法参数中即为验证参数对象。(添加在方法上无效)
@Validated
@RequestMapping("/books")
@RestController
public class BookController { @GetMapping("/test1")
public String test1(@NotBlank(message = "name不能为空") @Length(min = 2, max = 10 , message="name 长度必须在{min}-{max}之间") String name){
return "test1";
} @GetMapping("test2")
public String test2(@Validated Book book){
return "test2";
}
}
这样就可以了,在浏览器中输入 /books/test1/ 不输入参数会抛异常 /books/test1?name=sdf 则可以通过验证
/books/test2/ 异常 /books/test2?name=sdf&price=0.5就ok了


Spring Boot 还允许我们自定义Validatior
自定义注解
package com.spring.boot.utils; import javax.validation.Constraint;
import javax.validation.Payload;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; //@Target定义范围 在字段 参数上使用
@Target({ElementType.FIELD, ElementType.PARAMETER})
//定义可见范围 RUNTIME整个运行阶段都可见
@Retention(RetentionPolicy.RUNTIME)
//Constraint指定具体校验器类
@Constraint(validatedBy = DateTimeValidator.class)
//@interface 定义注解
public @interface DateTime {
String message() default "格式错误"; String format() default "yyyy-MM-dd"; Class<?>[] groups() default {}; Class<? extends Payload>[] payload() default {};
}
具体的校验器
package com.spring.boot.utils; import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;
import java.text.ParseException;
import java.text.SimpleDateFormat; /**
* 日期格式验证
*/
public class DateTimeValidator implements ConstraintValidator<DateTime, String> { private DateTime dateTime; @Override
//初始化,它可以获得当前注解的所有属性
public void initialize(DateTime constraintAnnotation) {
this.dateTime = constraintAnnotation;
} @Override
//约束验证的主体方法,其中s就是验证参数的具体事例,context代表约束执行的上下文
public boolean isValid(String s, ConstraintValidatorContext constraintValidatorContext) {
//如果为空不用验证, 为空验证可以用@NotBlank、@NotNull等参数进行控制
if (s == null) {
return true;
} String format = dateTime.format();
if (s.length() != format.length()) {
return false;
} SimpleDateFormat simpleDateFormat = new SimpleDateFormat(format);
try {
simpleDateFormat.parse(s);
} catch (ParseException e) {
return false;
}
return true;
}
}
然后在控制层使用他
@GetMapping("/test3")
public String test3(@NotBlank(message = "date 不能为空") @DateTime(message = "您如的格式错误,正确的格式为:{format}",format = "yyyy-MM-dd HH:mm") String date){
return "test3";
}
测试

分组验证
有时候我们需要对一个实体类有多种验证方式,在不同的情况下使用不同的验证方式,比如id,新增的时候是不需要的,更新时是必须的。
定义一个验证组,里面写上不同的空接口类即可
public class Groups {
public interface Update{
}
public interface Default{
}
}
实体类
groups属性的作用就让@Validated注解只验证与自身value属性想匹配的字段,可多个,只要满足就会去纳入验证范围。
public class Book {
@NotNull(message = "id 不能为空" , groups = Groups.Update.class)
private Integer id;
@NotBlank(message = "name 不能为空" , groups = Groups.Default.class)
private String name;
@NotNull(message = "price 不能为空" , groups = Groups.Default.class)
private BigDecimal price;
控制层
创建一个ValidateControoler类,然后定义好insert、update两个方法,由于insert方法并不关心id字段,所以这里的@Validate的value属性写成Groups.Default.class就可以了,而update方法需要Id,所以此处@Validated注解的value属性值就要写成Groups.Default.class,Groups.Update.Class。代表只要是这分组下的数据都需要进行数据有效性校验操作
@RestController
public class ValidateController {
//default中没有id
@GetMapping("/insert")
public String insert(@Validated(value=Groups.Default.class) Book book){
return "insert";
} //update中有id校验
@GetMapping("update")
public String update(@Validated(value={Groups.Default.class,Groups.Update.class}) Book book){
return "update";
} }



Spring Boot (31) 数据验证的更多相关文章
- Spring Boot的数据访问:CrudRepository接口的使用
示例 使用CrudRepository接口访问数据 创建一个新的Maven项目,命名为crudrepositorytest.按照Maven项目的规范,在src/main/下新建一个名为resource ...
- Kotlin + Spring Boot 请求参数验证
编写 Web 应用程序的时候,经常要做的事就是要对前端传回的数据进行简单的验证,比如是否非空.字符长度是否满足要求,邮箱格式是否正确等等.在 Spring Boot 中,可以使用 Bean Valid ...
- spring boot+jwt 权限验证
上周看了一下jwt以前公司的开发都是使用session共享的方法.现在主流的两种方式一种是把登录信息保存在服务器,另一种则是把信息保存在客户端.在使用session 存储的时候会遇到很多的问题,随着项 ...
- Spring Boot的数据访问 之Spring Boot + jpa的demo
1. 快速地创建一个项目,pom中选择如下 <?xml version="1.0" encoding="UTF-8"?> <project x ...
- Spring Boot与数据
SpringBoot 着眼于JavaEE! 不仅仅局限于 Mybatis .JDBC. Spring Data JPA Spring Data 项目的目的是为了简化构建基于 Spring 框架应用的数 ...
- Spring Boot 高效数据聚合之道
项目地址和示例代码: https://github.com/lvyahui8/spring-boot-data-aggregator 背景 接口开发是后端开发中最常见的场景, 可能是RESTFul接口 ...
- (8)Spring Boot 与数据访问
文章目录 简介 整合基本的JDBC与数据源 整合 druid 数据源 整合 mybatis 简介 对于数据访问层,无论是 SQL 还是 NOSQL ,Spring Boot 默认都采用整合 Sprin ...
- Spring Boot框架 - 数据访问 - 整合Mybatis
一.新建Spring Boot项目 注意:创建的时候勾选Mybatis依赖,pom文件如下 <dependency> <groupId>org.mybatis.spring.b ...
- Spring Boot实现数据访问计数器
1.数据访问计数器 在Spring Boot项目中,有时需要数据访问计数器.大致有下列三种情形: 1)纯计数:如登录的密码错误计数,超过门限N次,则表示计数器满,此时可进行下一步处理,如锁定该账户 ...
随机推荐
- UVA 10564_ Paths through the Hourglass
题意: 由0-9的数字组成一个形如沙漏的图形,要求从第一行开始沿左下或者右下到达最后一行,问有多少种不同的路径,使最后路径上的整数之和为给定的某个数. 分析: 简单计数dp,从最后一行开始,设dp[i ...
- SQLSERVER数据库管理员的专用连接DAC
出处: http://www.cnblogs.com/lyhabc/archive/2012/09/23/2698702.html DAC:Dedicated Admin Connection 当SQ ...
- mssql存储过程异常处理
MSSQL2000和MSSQL2005以上版本的异常处理语法是不相同的. SQL Server 2005以上版本支持结构化异常处理,而MSSQL2000是不支持的. 1)先看MSSQL 2000的异常 ...
- 几种常见排序算法的java实现
一.几种常见的排序算法性能比較 排序算法 最好时间 平均时间 最坏时间 辅助内存 稳定性 备注 简单选择排序 O(n^2) O(n^2) O(n^2) O(1) 不稳定 n小时较好 直接插入排序 O( ...
- Windows下安Mac
Windows PC下安装苹果系统 第一步: 準備2個新邏輯分區,一個6G(os),一個隨意(Mac),且不要格式化. 第二步: 启动硬盘助手,选择下载好的苹果镜像文件 .再选择6G(os)分區,寫 ...
- 神马都是浮云,unity中自己写Coroutine协程源代码
孙广东 2014.7.19 无意之间看到了,Unity维基上的一篇文章, 是关于自己写协程的介绍. 认为非常好,这样能更好的了解到协程的执行机制等特性.还是不错的. 原文链接地址例如以下: ht ...
- [WebGL入门]五,矩阵的基础知识
注:文章译自http://wgld.org/,原作者杉本雅広(doxas),文章中假设有我的额外说明.我会加上[lufy:],另外.鄙人webgl研究还不够深入,一些专业词语,假设翻译有误,欢迎大家指 ...
- jmeter获取时间_time 函数
原始时间戳13位精确到毫秒:${__time(,)} 时间戳精确到秒10位:${__time(/1000,)} 时间日期到年月日2019-04-21:${__time(yyyy-MM-dd,)} 时间 ...
- frame pointer及其用途
1 什么是frame pointer frame pointer指向本函数栈帧顶,通过它可以找到本函数在进程栈中的位置.有专门的寄存器保存该值. 2 frame pointer有什么用 主要是back ...
- mysqld 与 python 邮件监控脚本 内存消耗对比
top - 21:38:40 up 1 day, 10:38, 5 users, load average: 0.00, 0.01, 0.17Tasks: 88 total, 1 running, 8 ...