学习Spring Boot:(十)使用hibernate validation完成数据后端校验

前言

后台数据的校验也是开发中比较注重的一点,用来校验数据的正确性,以免一些非法的数据破坏系统,或者进入数据库,造成数据污染,由于数据检验可能应用到很多层面,所以系统对数据校验要求比较严格且追求可变性及效率。

了解

了解一点概念性的东东。
* JSR 303 是 Java 为 Bean 数据合法性校验提供的标准框架,它已经包含在 JavaEE 6.0 中 。
* Hibernate Validator 是 JSR 303 的一个参考实现,所以它多实现了几个校验规则。
* Spring 4.0 拥有自己独立的数据校验框架,同时支持 JSR303 标准的校验框架。
* 在已经标注了 JSR303 注解的表单/命令对象前标注一个@Valid,Spring MVC 框架在将请求参数绑定到该入参对象后,就会调用校验框架根据注解声明的校验规则实施校验
* Spring MVC 是通过对处理方法签名的规约来保存校验结果的:前一个表单/命令对象的校验结果保存到随后的入参中,这个保存校验结果的入参必须是 BindingResult 或Errors 类型,这两个类都位于org.springframework.validation 包中。
* 需校验的 Bean 对象和其绑定结果对象或错误对象时成对出现的,它们之间不允许声明其他的入参
* Errors 接口提供了获取错误信息的方法,如 getErrorCount() 或getFieldErrors(String field)
* BindingResult 扩展了 Errors 接口。

支持的注解

JSR 提供的校验注解:

@Null   被的注解元素必须为 null    
@NotNull    被注解的元素必须不为 null    
@AssertTrue     被注解的元素必须为 true    
@AssertFalse    被注解的元素必须为 false    
@Min(value)     被注解的元素必须是一个数字,其值必须大于等于指定的最小值    
@Max(value)     被注解的元素必须是一个数字,其值必须小于等于指定的最大值    
@DecimalMin(value)  被注解的元素必须是一个数字,其值必须大于等于指定的最小值    
@DecimalMax(value)  被注解的元素必须是一个数字,其值必须小于等于指定的最大值    
@Size(max=, min=)   被注解的元素的大小必须在指定的范围内   集合或数组 集合或数组的大小是否在指定范围内 
@Digits (integer, fraction)     被注解的元素必须是一个数字,验证是否是符合指定格式的数字,interger指定整数精度,fraction指定小数精度。   
@Past   被注解的元素必须是一个过去的日期    
@Future     被注解的元素必须是一个将来的日期    
@Pattern(regex=,flag=)  被注解的元素必须符合指定的正则表达式

Hibernate Validator 提供的校验注解:

@NotBlank(message =)   验证字符串非null,且长度必须大于0    
@Email  被注释的元素必须是电子邮箱地址    
@Length(min=,max=)  被注解的值大小必须在指定的范围内    
@NotEmpty   被注解的字符串的必须非空    
@Range(min=,max=,message=)  验证该值必须在合适的范围内

可以在需要验证的属性上,使用多个验证方式,它们同时生效。
spring boot web 已经有 hibernate-validation 的依赖,所以不需要再手动添加依赖。

使用

首先我在我的实体类上写了几个校验注解。

public class SysUserEntity implements Serializable {private static final long serialVersionUID = 1L;//主键private Long id;//用户名@NotBlank(message = "用户名不能为空", groups = {AddGroup.class, UpdateGroup.class})private String username;//密码@NotBlank(message = "密码不能为空", groups = {AddGroup.class})private String password;//手机号@Pattern(regexp = "^1([345789])\\d{9}$",message = "手机号码格式错误")@NotBlank(message = "手机号码不能为空")private String mobile;//邮箱@Email(message = "邮箱格式不正确")private String email;//创建者private Long createUserId;//创建时间private Date createDate;
// ignore set and get

使用@Validated进行校验

首先了解下:
关于@Valid和@Validated的区别联系
* @Valid: javax.validation, 是javax,也是就是jsr303中定义的规范注解
* @Validated: org.springframework.validation.annotation, 是spring自己封装的注解。参数校验失败抛出 org.springframework.validation.BindException 异常。

@Validated@Valid的一个变种,扩展了@Valid的功能,支持group分组校验的写法,所以为了校验统一,尽量使用@Validated

在controller自定义一个接口

@PostMapping("/valid")public ResponseEntity<String> valid(@Validated @RequestBody SysUserEntity user, BindingResult result) {if (result.hasErrors()) {return ResponseEntity.status(BAD_REQUEST).body("校验失败");}return ResponseEntity.status(OK).body("校验成功");}

需要注意的有几点:
* 需要校验对象的时候,需要加上 spring 的校验注解 @Validated ,表示我们需要 spring 对它进行校验,而校验的信息会存放到其后的BindingResult中。
* BindingResult 必须和检验对象紧邻,中间不能穿插任何参数,如果有多个校验对象 @Validated @RequestBody SysUserEntity user, BindingResult result, @Validated @RequestBody SysUserEntity user1, BindingResult result1

我在前端用 Swagger 进行测试下。
我发送一个 body,将 手机号输错:

{"createDate": "","createUserId": 0,"email": "k@wuwii.com","id": 0,"mobile": "12354354","password": "123","username": "12312"
}

后端调试下 BindingResult 的结果,发现结果:
image
只要注意下 errors 属性,它是校验所有不符合规则的,是一个数组。

分组校验

有时候 ,我们在新增和更新的时候校验效果是不一样的。例如上面,我在User新增的时候需要判断密码是不是为空,但是更新的时候我不做校验。这个时候就也要用到分组校验了。

@NotBlank(message = "密码不能为空", groups = {AddGroup.class})
private String password;

将Contoller中的校验修改下。

(@Validated({AddGroup.class}) @RequestBody SysUserEntity user, BindingResult result)

上面的意思是只有分组是AddGroup的校验才生效,其余的校验忽略。

经过我测试,把分组情况分下:

  1. 在controller校验没加分组的时候,只对实体类的没有分组的注解有效。
  2. 在controller校验加分组的时候,只对实体类的当前分组的注解有效,没有注解的也无效。
  3. 当校验有两个分组的时候@Validated({AddGroup.class, UpdateGroup.class}),满足当前两个分组其中任意一个都可以校验,两个注解同时一起出现,也没问题,而且检验不通过的信息不会重复。

自定义校验

有时候系统提供给我们的校验注解,并不够用,我们可以自定义校验,来满足我们的业务需求。

例如:现在我们有一个需求,需要检测一条信息的敏感词汇,如sb ……文明人,举个栗子 ……

自定义校验注解
// 注解可以用在哪些地方
@Target({METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER})
@Retention(RUNTIME)
@Documented
// 指定校验规则实现类
@Constraint(validatedBy = {NotHaveSBValidator.class})
public @interface NotHaveSB {//默认错误消息String message() default "不能包含字符sb";//分组Class<?>[] groups() default {};//负载Class<? extends Payload>[] payload() default {};//指定多个时使用@Target({FIELD, METHOD, PARAMETER, ANNOTATION_TYPE})@Retention(RUNTIME)@Documented@interface List {NotHaveSB[] value();}}
规则校验实现类
// 可以指定检验类型,这里选择的是 String
public class NotHaveSBValidator implements ConstraintValidator<NotHaveSB, String> {@Overridepublic void initialize(NotHaveSB notHaveSB) {}/**** @param s 待检验对象* @param constraintValidatorContext 检验上下文,可以设置检验的错误信息* @return false 代表检验失败*/@Overridepublic boolean isValid(String s, ConstraintValidatorContext constraintValidatorContext) {return !StringUtils.isNotBlank(s) || !s.toLowerCase().contains("sb");}
}

所有的验证者都需要实现ConstraintValidator接口,它的接口也很形象,包含一个初始化事件方法,和一个判断是否合法的方法。

测试一下喂

现在我的用户类上,也没什么多余的字段拿出来测试,暂时把 password 字段拿来测试吧。

//@NotBlank(message = "密码不能为空", groups = AddGroup.class)@NotHaveSBprivate String password;

image

手动校验

这个是我最终想要的处理方式。
由于现在都是前后端分离开发的,校验失败的时候,抛出自定义的异常,然后统一处理这些异常,最后将相关的错误提示信息返回给前端处理。

新建一个验证工具类

public class ValidatorUtils {private static Validator validator;static {validator = Validation.buildDefaultValidatorFactory().getValidator();}/*** 手动校验对象** @param object 待校验对象* @param groups 待校验的组* @throws KCException 校验不通过,则抛出 KCException 异常*/public static void validateEntity(Object object, Class<?>... groups)throws KCException {Set<ConstraintViolation<Object>> constraintViolations = validator.validate(object, groups);if (!constraintViolations.isEmpty()) {String msg = constraintViolations.parallelStream().map(ConstraintViolation::getMessage).collect(Collectors.joining(","));throw new KCException(msg);}}
}

它主要做的事情就是验证我们的待验证对象,验证不同通过的时候,抛出自定义异常,在后台统一处理异常就可以了。

在业务中直接调用就可以了,有分组添加分组就行

@PostMapping("/valid1")public ResponseEntity<String> customValid(@RequestBody SysUserEntity user) {ValidatorUtils.validateEntity(user);return ResponseEntity.status(OK).body("校验成功");}

最后测试一下,查看返回结果是否符合预期:

image

手动校验的补充

决定还是采用注解的形式进行编码,本来想用处理方法参数的装配进行检验,写好了发现和 @responseBody 不能同时使用,然后发现还是可以使用 @Validated 直接校验,抛出异常, 进行捕捉异常统一处理。

    @PostMapping()@ApiOperation("新增")public ResponseEntity insert(@Validated SysUserAddForm user) 

在全局异常处理里面加上 处理绑定参数异常 org.springframework.validation.BindException

    /*** 参数检验违反约束(数据校验)* @param e BindException* @return error message*/@org.springframework.web.bind.annotation.ExceptionHandler(BindException.class)public ResponseEntity<String> handleConstraintViolationException(BindException e) {LOGGER.debug(e.getMessage(), e);return ResponseEntity.status(BAD_REQUEST).body(e.getBindingResult().getAllErrors().stream().map(DefaultMessageSourceResolvable::getDefaultMessage).collect(Collectors.joining(",")));}

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

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

相关文章

js面向对象与java面向对象的区别,被坑了,js语法跟Java面向对象语法还是有区别的...

请见代码&#xff0c;实现功能是要点表格当前行里的删除链接&#xff0c;直接删除当前行&#xff0c;并且删到最后一行的时候&#xff0c;把包含当前table的div直接隐藏了。function delCurrentTr(){//这两句一定要写在$(this)对象删除之前&#xff0c;否则$(this)对象执行删除…

学习Spring Boot:(十一) 自定义装配参数

前言 SpringMVC 中 Controller 中方法的参数非常灵活&#xff0c;得益于它的强大自动装配&#xff0c;这次将根据上次遗留下的问题&#xff0c;将研究下装配参数。 正文 SpringMVC中使用了两个接口来处理参数&#xff1a; * HandlerMethodArgumentResolver 处理方法请求参数…

php多个参数绑定,php – 如何绑定多个参数到MySQLi查询

我有一个mysql查询,但我不能绑定paramSELECT users.email,users.handle,userprofile.mobile FROM users,userprofile WHERE users.email ? OR users.handle ? OR userprofile.mobile?我试过下面的行$query "SELECT users.email,users.handle,userprofile.mobile FROM …

学习Spring Boot:(十二)Mybatis 中自定义枚举转换器

前言 在 Spring Boot 中使用 Mybatis 中遇到了字段为枚举类型&#xff0c;数据库存储的是枚举的值&#xff0c;发现它不能自动装载。 解决 内置枚举转换器 MyBatis内置了两个枚举转换器分别是&#xff1a;org.apache.ibatis.type.EnumTypeHandler 和 org.apache.ibatis.typ…

php旅行社网站源码在线支付,PHP各大支付平台在线支付集成源码

演示查看&#xff1a;http://www.erdangjiade.com/ph...php给客户开发网站的时候需要用到各大平台付款功能&#xff0c;下面就免费分享给大家&#xff0c;此类是个成熟类&#xff0c;网上down下来的&#xff0c;经过修改测试了(可以直接拿来使用&#xff0c;附带使用方法&#…

学习Spring Boot:(十三)配置 Shiro 权限认证

经过前面学习 Apache Shiro &#xff0c;现在结合 Spring Boot 使用在项目里&#xff0c;进行相关配置。 正文 添加依赖 在 pom.xml 文件中添加 shiro-spring 的依赖&#xff1a; <dependency><groupId>org.apache.shiro</groupId><artifactId>shir…

php设计之初用于什么,PHP设计模式(七)之门面模式

一、什么是外观模式(Facade Pattern)定义&#xff1a;外观模式又称门面模式&#xff0c;提供一个统一的接口&#xff0c;用来访问子系统中的一群接口。外部与子系统之间的通信采用门面(Facade)对象来完成。【举例】比如麦当劳套餐&#xff0c;套餐包含鸡肉卷、汉堡包、可乐等N个…

学习Spring Boot:(十四)spring-shiro的密码加密

前言 前面配置了怎么使用 shiro &#xff0c;这次研究下怎么使用spring shiro的密码加密&#xff0c;并且需要在新增、更新用户的时候&#xff0c;实现生成盐&#xff0c;加密后的密码进行入库操作。 正文 配置凭证匹配器 Beanpublic HashedCredentialsMatcher hashedCreden…

php 仿高德,仿高德路线规划滑动效果

因为项目有个界面要模仿高德地图路径规划滑动效果&#xff0c;因此写了demo&#xff0c;并简单说下分析过程。高德地图效果演示:仿高德路线规划滑动.gifdemo效果演示:高德地图规划滑动.gif一. 分析首先&#xff0c;我们可以看出这个滚动的视图应该是UIScrollView或者UIScrollVi…

php验证码完整功能,PHP验证码功能的实现

/***产生验证码图片*/public function actionVerfiycode() {Header ( "Content-type: image/gif" );$border 0; //是否要边框 1要:0不要$how 4; //验证码位数$w $how * 15; //图片宽度$h 20; //图片高度$fontsize 5; //字体大小$alpha "abcdefghijkmnopqr…

学习Spring Boot:(十五)使用Lombok来优雅的编码

前言 Lombok 是一种 Java? 实用工具&#xff0c;可用来帮助开发人员消除 Java 的冗长&#xff0c;尤其是对于简单的 Java 对象&#xff08;POJO&#xff09;。它通过注解实现这一目的。 正文 添加依赖 在 pom.xml 文件中添加相关依赖&#xff1a; <lombok.version>1.…

java 品尝饮料,java细节经典题型

28. 选项中哪一行代码可以替换题目中//add code here 而不产生编译错误?() [java] view plaincopy 1. public abstract class MyClass { 2. 3. 4. 5.......Java 基础试题 一:选择题(1*3030) (题目写在答题纸上面) 1:Java 提供哪几种运算符多选 ( abcd )。 A)算术运算符 B)位运…

学习Spring Boot:(十六)使用Shiro与JWT 实现认证服务

前言 代码可以参考 需要把Web应用做成无状态的&#xff0c;即服务器端无状态&#xff0c;就是说服务器端不会存储像会话这种东西&#xff0c;而是每次请求时access_token进行资源访问。这里我们将使用 JWT 1&#xff0c;基于散列的消息认证码&#xff0c;使用一个密钥和一个消…

java泛型和注解,泛型 · 注解和泛型 · 看云

[TOC]# 泛型## 为什么要使用泛型在之前学过的集合框架中&#xff0c;List和Map都使用了泛型技术来确认其内容的数据类型。如果不使用泛型&#xff0c;在程序运行阶段&#xff0c;会带来数据类型转型的错误风险。~~~List list new ArrayList();list.add("tom");for (…

java程序单引号报错,javapoigetInpuStream报错br/是这样的, 爱问知识人

是这样的&#xff0c;我写了重载了两个getInputStream方法&#xff0c;当调用不带参数的方法时&#xff0c;运行正常&#xff0c;当调用带参的方法则报 Can not find a java.io.InputStream with the name [inputStream] in是这样的&#xff0c;我写了重载了两个getInputStream…

学习Spring Boot:(十七)Spring Boot 中使用 Redis

前言 Redis是一个由Salvatore Sanfilippo写的key-value存储系统。 edis是一个开源的使用ANSI C语言编写、遵守BSD协议、支持网络、可基于内存亦可持久化的日志型、Key-Value数据库&#xff0c;并提供多种语言的API。 通常被称为数据结构服务器&#xff0c;因为值&#xff08;va…

jqueryd登录异步请求 java,ajaxd的js和jquery实现

先来看一下javascript的。var httpxml;httpxml new XMLHttpRequest();httpxml.onreadystatechangefunction(){ //当服务器响应就绪时执行函数(就是说服务器准备好了你可以发请求了)if(httpxml.status 200 && httpxml.readyState 4){//这里200和4代表响应的状态&…

学习Spring Boot:(十八)Spring Boot 中session共享

前言 前面我们将 Redis 集成到工程中来了&#xff0c;现在需要用它来做点实事了。这次为了解决分布式系统中的 session 共享的问题&#xff0c;将 session 托管到 Redis。 正文 引入依赖 除了上篇文章中引入 spring-boot-starter-data-redis&#xff0c;还需要 spring-sess…

matlab 码元扩展,扩频通信及matlab仿真

扩展频谱通信以及直接扩频的matlab仿真号无关)扩展频谱后成为宽频带信号&#xff0c;然后再进行传输的一种系统。待传输的基带信号就是信源发出的数字信号。特定的扩频函数通常选用各种伪随机序列(扩频码)&#xff0c;其码元传输速率远大于基带信号速率&#xff0c;因而和基带信…

学习Spring Boot:(十九)Shiro 中使用缓存

前言 在 shiro 中每次去拦截请求进行权限认证的时候&#xff0c;都会去数据库查询该用户的所有权限信息&#xff0c; 这个时候就是有一个问题了&#xff0c;因为用户的权限信息在短时间内是不可变的&#xff0c;每次查询出来的数据其实都是重复数据&#xff0c;没必要每次都去…