Springboot学习参数校验(五)-1

使用hibernate validator出现javax.validation.UnexpectedTypeException: HV000030: No validator could be found for constraint的错误, 需要 注意

@NotNull 和 @NotEmpty 和@NotBlank 区别

@NotEmpty 用在集合类上面
@NotBlank 用在String上面
@NotNull 用在基本类型上

作为服务端开发,验证前端传入的参数的合法性是一个必不可少的步骤,但是验证参数是一个基本上是一个体力活,而且冗余代码繁多,也影响代码的可阅读性,所以有没有一个比较优雅的方式来解决这个问题?

这么简单的问题当然早就有大神遇到并且解决了,这一篇文章主要讲一下解决基于spring-boot的验证参数的比较好的方法:利用validator-api来进行验证参数。

在spring-boot-starter-web包里面有hibernate-validator包,它提供了一系列验证各种参数的方法,所以说spring-boot已经帮我们想好要怎么解决这个问题了。

(1) 入门例子;

(2) 国际化;

(3) 在代码中添加错误信息;

(1) 入门例子;

   Validator主要是校验用户提交的数据的合理性的,比如是否为空了,密码长度是否大于6位,是否是纯数字的,等等。那么在spring boot怎么使用这么强大的校验框架呢。

在这里我们主要是使用注解进行学习。我们先说说我们的需求:

我们有一个demo.html,在页面上有两个元素 姓名输入框,密码输入库,提交按钮。

提交到后台之后,使用Validator进行校验,然后如果存在错误,转发到demo.html,

我们先编写一个实体类接收用户的输入,以及使用Validator注解校验:

package com.kfit.demo;

import org.hibernate.validator.constraints.Length;
import org.hibernate.validator.constraints.NotEmpty;

public class Demo {

private long id;

@NotEmpty(message="姓名不能为空")
private String name;

@NotEmpty(message="密码不能为空")
@Length(min=6,message="密码长度不能小于6位")
private String password;

publiclong getId() {
   return id;
}

publicvoid setId(longid) {
   this.id = id;
}

public String getName() {
   return name;
}

public void setName(String name) {
   this.name = name;
}

public String getPassword() {
   return password;
}

public void setPassword(String password) {
   this.password = password;
}

@Override
public String toString() {
   return "Demo [id=" + id + ", name=" + name + ", password=" + password + "]";
}

}
这个实体类在属性上加入了注解@NotEmpty,@Length,那么常用的注解有:

约束注解名称 约束注解说明
@null 验证对象是否为空
@notnull 验证对象是否为非空
@asserttrue 验证 boolean 对象是否为 true
@assertfalse 验证 boolean 对象是否为 false
@min 验证 number 和 string 对象是否大等于指定的值
@max 验证 number 和 string 对象是否小等于指定的值
@decimalmin 验证 number 和 string 对象是否大等于指定的值,小数存在精度
@decimalmax 验证 number 和 string 对象是否小等于指定的值,小数存在精度
@size 验证对象(array,collection,map,string)长度是否在给定的范围之内
@digits 验证 number 和 string 的构成是否合法
@past 验证 date 和 calendar 对象是否在当前时间之前
@future 验证 date 和 calendar 对象是否在当前时间之后
@pattern 验证 string 对象是否符合正则表达式的规则

@Email 验证邮箱

实际例子:

@size (min=3, max=20, message=”用户名长度只能在3-20之间”)

@size (min=6, max=20, message=”密码长度只能在6-20之间”)

@pattern (regexp=”[a-za-z0-9._%+-]+@[a-za-z0-9.-]+\.[a-za-z]{2,4}”, message=”邮件格式错误”)

@Length(min = 5, max = 20, message = “用户名长度必须位于5到20之间”)

@Email(message = “比如输入正确的邮箱”)

@NotNull(message = “用户名称不能为空”)
@Max(value = 100, message = “年龄不能大于100岁”)
@Min(value= 18 ,message= “必须年满18岁!” )
@AssertTrue(message = “bln4 must is true”)

@AssertFalse(message = “blnf must is falase”)
@DecimalMax(value=”100”,message=”decim最大值是100”)
DecimalMin(value=”100”,message=”decim最小值是100”)
@NotNull(message = “身份证不能为空”)
@Pattern(regexp=”^(\d{18,18}|\d{15,15}|(\d{17,17}[x|X]))$”, message=”身份证格式错误”)

好了,这个不是我们这节的重点,这里简单说一下而已,那么之后我们应该怎么做呢?我们需要编写一个Controller进行访问的时候,能访问到demo.html已经点击提交按钮的处理方法,具体看如下代码:

@RequestMapping("/demo")
public String demo(Model model){
   model.addAttribute("demo",new Demo());
   return "/demo";
}


@RequestMapping("/demoAdd")
public String demoAdd(@Valid Demo demo,BindingResult result,Model model){
   //有错误信息.
   model.addAttribute("demo",demo);
   if(result.hasErrors()){
       List<ObjectError>  list = result.getAllErrors();
       for(ObjectError  error:list){
       System.out.println(error.getCode()+"---"+error.getArguments()+"---"+error.getDefaultMessage());
       }

       return "demo";
   }
   return "/demo";
}

这里的代码还是需要简单说明下,我们使用@Valid指定要校验的实体类。

BindingResult 所有的错误信息都会保存在这个类中,我们可以使用result.hasErrors() 判断是否有错误信息,有的话,我么转发到我们原先的访问的hello.html,如果没有的话,我们正常应该是调整到list.html之类的,这里只是为了方便测试跳回了demo.html,但是如果没有任何错误信息的话,那么在页面上是不会显示错误信息的。

   好了,接下里我们看看demo.html是怎么编写的吧?





hello spring boot

<form action="/demoAdd" method="post" th:object="${demo}">
   <p>姓名:<input type="text" name="name" th:value="*{name}" />
   </p>
   <p th:if="${#fields.hasErrors('name')}" th:errors="*{name}">Name Error</p>
   <p>密码:<input type="text" name="password" th:value="*{password}"  />
   </p>
   <p th:if="${#fields.hasErrors('password')}" th:errors="*{password}">password Error</p>
   <p><button>提交</button></p>
</form>


这里我们使用的thymeleaf进行展示数据的,使用jsp的代码需要用到tag标签,也能实现相同的效果,自行百度学习。这里核心代码就是:

Name Error

这句代码一旦后台有返回异常信息的话,就会显示name对应的message,如果满足了两个条件的话,是会返回两个的,之间是用
进行处理的,就如我们的password什么都不填写的情况下是会显示如下信息的:

密码:
密码不能为空
密码长度不能小于6位

(2) 国际化;

   在上一节我们就讲过国际化了,那么如何在Validator加入国际化呢,很简单的,只需要在国际化配置文件加入相应的配置如:

demo.name = name is not empty.

这里需要注意的地方是:

必须放在classes目录下,而且必须用ValidationMessages这个名字

也就是文件名称需要命令为:

ValidationMessages.properties

ValidationMessages_en.properties

那么修改Demo.java文件:

@NotEmpty(message=”{demo.name}”)

(3) 在代码中添加错误信息;

   有些代码是很难使用Validator的注解来实现的,那么我们怎们在返回的信息添加我们自己的判断呢,比如我们现在要求用户输入的name不能重复,那么势必我们会这么一段代码: 如果存在name,那么返回“该name已经存在了”。其实这个也是很简单,只需要一句话代码就可以添加自定义错误字段的信息了:

result.rejectValue(“name”, “misFormat”, “这个name已经注册过了!”);

当然在添加的时候,外层应该有一个if(isExist(“name”))这样的代码,这里没有进行编写,就直接添加了,实际开发请自行从数据库获取,然后进行判断

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值