博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Spring 校验器(Validator)
阅读量:6447 次
发布时间:2019-06-23

本文共 5375 字,大约阅读时间需要 17 分钟。

Spring校验器,参数校验从此简单。

img_e1f20e9a762037a48a3c4bd0a2b89b14.png
image.png

应用在执行业务逻辑之前,必须通过校验保证接受到的输入数据是合法正确的,但很多时候同样的校验出现了多次,在不同的层,不同的方法上,导致代码冗余,浪费时间,违反DRY原则。

  1. 每一个控制器都要校验
  2. 过多的校验参数会导致代码太长
  3. 代码的复用率太差,同样的代码如果出现多次,在业务越来越复杂的情况下,维护成本呈指数上升。

可以考虑把校验的代码封装起来,来解决出现的这些问题。

JSR-303

JSR-303是Java为Bean数据合法性校验提供的标准框架,它定义了一套可标注在成员变量,属性方法上的校验注解。

Hibernate Validation提供了这套标准的实现,在我们引入Spring Boot web starter或者Spring boot starter validation的时候,默认会引入Hibernate Validation。

用法实例

说了这么多废话,上代码。

  1. 引入SpringBoot项目
org.springframework.boot
spring-boot-starter-web
org.hibernate.validator
hibernate-validator
org.projectlombok
lombok
  1. 编写校验对象
@Datapublic class User {    // 名字不允许为空,并且名字的长度在2位到30位之间    // 如果名字的长度校验不通过,那么提示错误信息    @NotNull    @Size(min=2, max=30,message = "请检查名字的长度是否有问题")    private String name;    // 不允许为空,并且年龄的最小值为18    @NotNull    @Min(18)    private Integer age;}
  1. 创建控制器
@SpringBootApplication@RestControllerpublic class UserApplication{    public static void main(String[] args) {        SpringApplication.run(UserApplication.class,args);    }        // 1. 要校验的参数前,加上@Valid注解    // 2. 紧随其后的,跟上一个BindingResult来存储校验信息    @RequestMapping("/test1")    public Object test1(            @Valid User user,            BindingResult bindingResult    ) {        //如果检验出了问题,就返回错误信息        // 这里我们返回的是全部的错误信息,实际中可根据bindingResult的方法根据需要返回自定义的信息。        // 通常的解决方案为:JSR-303 + 全局ExceptionHandler        if (bindingResult.hasErrors()){            return bindingResult.getAllErrors();        }        return "OK";    }    }
  1. 运行应用

稍作演示下运行的结果,可以看出校验框架已经生效了。

img_6ec8baad35d42dea41b50a3e3eb9e1d4.png
校验年龄
img_28274848a5228d2945204182f447412e.png
校验名称
img_115d98e7940d96a583a5ddf96ec226e7.png
校验通过

常见的校验注解

@Null 被注释的元素必须为 null

@NotNull 被注释的元素必须不为 null
@AssertTrue 被注释的元素必须为 true
@AssertFalse 被注释的元素必须为 false
@Min(value) 被注释的元素必须是一个数字,其值必须大于等于指定的最小值
@Max(value) 被注释的元素必须是一个数字,其值必须小于等于指定的最大值
@DecimalMin(value) 被注释的元素必须是一个数字,其值必须大于等于指定的最小值
@DecimalMax(value) 被注释的元素必须是一个数字,其值必须小于等于指定的最大值
@Size(max=, min=) 被注释的元素的大小必须在指定的范围内
@Digits (integer, fraction) 被注释的元素必须是一个数字,其值必须在可接受的范围内
@Past 被注释的元素必须是一个过去的日期
@Future 被注释的元素必须是一个将来的日期
@Pattern(regex=,flag=) 被注释的元素必须符合指定的正则表达式

Hibernate Validator提供的校验注解:

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

自定义校验注解

有时候,第三方库中并没有我们想要的校验类型,好在系统提供了很好的扩展能力,我们可以自定义检验。

比如,我们想校验用户的手机格式,写手机号码校验器

  1. 编写校验注解
// 我们可以直接拷贝系统内的注解如@Min,复制到我们新的注解中,然后根据需要修改。@Target({METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER})@Retention(RUNTIME)@Documented//注解的实现类。@Constraint(validatedBy = {IsMobileValidator.class})public @interface IsMobile {    //校验错误的默认信息    String message() default "手机号码格式有问题";    //是否强制校验    boolean isRequired() default false;        Class
[] groups() default {}; Class
[] payload() default {};}
  1. 编写具体的实现类
    我们知道注解只是一个标记,真正的逻辑还要在特定的类中实现,上一步的注解指定了实现校验功能的类为IsMobileValidator。
// 自定义注解一定要实现ConstraintValidator接口奥,里面的两个参数// 第一个为 具体要校验的注解// 第二个为 校验的参数类型public class IsMobileValidator implements ConstraintValidator
{ private boolean required = false; private static final Pattern mobile_pattern = Pattern.compile("1\\d{10}"); //工具方法,判断是否是手机号 public static boolean isMobile(String src) { if (StringUtils.isEmpty(src)) { return false; } Matcher m = mobile_pattern.matcher(src); return m.matches(); } @Override public void initialize(IsMobile constraintAnnotation) { required = constraintAnnotation.isRequired(); } @Override public boolean isValid(String phone, ConstraintValidatorContext constraintValidatorContext) { //是否为手机号的实现 if (required) { return isMobile(phone); } else { if (StringUtils.isEmpty(phone)) { return true; } else { return isMobile(phone); } } } }
  1. 测试自定义注解的功能
@Datapublic class User {    @NotNull    @Size(min=2, max=30,message = "请检查名字的长度是否有问题")    private String name;    @NotNull    @Min(18)    private Integer age;    //这里是新添加的注解奥    @IsMobile    private String phone;}
  1. 测试

    img_e05b3d7b1cca0c6ac3abcb915bbe0566.png
    通过
img_2203084a510da4bb1ce56a1536f62438.png
手机号有问题

可以看出自定义的注解已经生效了。

我们还可以继续优化的地方,新建一个全局的异常,如果校验失败的话,抛出全局的业务异常,捕获业务异常,然后返回用户友好的提示信息。

额外

也可以通过方法的校验。

  1. 控制器上添加@Validated注解
  2. 在控制器的方法上添加校验注解,@Min,@Max等。
@Validated@RestController@SpringBootApplicationpublic class UserApplication{    public static void main(String[] args) {        SpringApplication.run(UserApplication.class,args);    }    @RequestMapping("/test2")    public String test2(            @IsMobile String phone    ){        return phone + "ok";    }    @ExceptionHandler(ConstraintViolationException.class)    @ResponseBody    public Object handleConstraintViolationException(ConstraintViolationException cve){        HashSet
messageSet = new HashSet(); for (ConstraintViolation constraintViolation : cve.getConstraintViolations()) { messageSet.add(constraintViolation.getMessage()); } return messageSet; }}
img_9217539eed3c6156d2b37de135cb5e34.png
类的校验规则

最后

通过使用校验器,所有的控制器,我们都不用再去做校验啦,代码再回看是不是清爽很多。我们写代码很简答,但是一定要想到如何把代码写的更简单,更清晰,更利于维护,写重复的代码是在浪费自己的时间奥。

以后再碰到参数校验的情况,首先想到的不是直接就去校验,可以查找自己是否写过某一类的验证器,可以直接拿来即用。

希望能帮助大家。

另外

我在知识星球开通了一个交流群,免费的。希望能和大家一起交流学习,进群的可以获得500G Java学习资料奥,多年的收藏,除了Java之外,还收藏了IOS,Linux,Android相关资料。想要的也可以找我拿。

img_f7cccc702db1f653da22ca4bad0d1171.png
猩球图片

转载地址:http://hcvwo.baihongyu.com/

你可能感兴趣的文章
open-falcon
查看>>
周会会议2018.4.20
查看>>
二叉树的建立与遍历
查看>>
三菱plc输出指示灯不亮怎么办(转载)
查看>>
doc2vec使用说明(一)gensim工具包TaggedLineDocument
查看>>
App测试中ios和Android的区别
查看>>
java.lang.NullPointerException&com.cb.action.LoginAction.execute(LoginAction.java:48)
查看>>
理解Docker :Docker 网络
查看>>
通过Application存取公共数据比如登录信息等..
查看>>
intellij maven配置与使用
查看>>
SpringMVC文件下载与JSON格式
查看>>
Q:图像太大,在opencv上显示不完全
查看>>
修正锚点跳转位置 避免头部fixed固定部分遮挡
查看>>
Dubbo序列化多个CopyOnWriteArrayList对象变成同一对象的一个大坑!!
查看>>
linux下ping不通的解决方法
查看>>
利用ItextPdf、core-renderer-R8 来生成PDF
查看>>
irc操作小记
查看>>
JAVA 与 PHP 的不同和相同
查看>>
03-Java环境变量配置
查看>>
Python mysql操作
查看>>