点击上方蓝色“程序猿DD”,选择“设为星标”

回复“资源”获取独家整理的学习资料!

来源 | 公众号「乱敲代码」

做web开发有一点很烦人就是要校验参数,基本上每个接口都要对参数进行校验,比如一些格式校验 非空校验都是必不可少的。如果参数比较少的话还是容易 处理的一但参数比较多了的话代码中就会出现大量的IF ELSE就比如下面这样:

这个例子只是校验了一下空参数。如果需要验证邮箱格式和手机号格式校验的话代码会更多,所以介绍一下validator通过注解的方式进行校验参数。

什么是Validator

Bean Validation是Java定义的一套基于注解的数据校验规范,目前已经从JSR 303的1.0版本升级到JSR 349的1.1版本,再到JSR 380的2.0版本(2.0完成于2017.08),已经经历了三个版本 。在SpringBoot中已经集成在 starter-web中,所以无需在添加其他依赖。

注解介绍

validator内置注解

注解 详细信息
@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(value) 被注释的元素必须符合指定的正则表达式

Hibernate Validator 附加的 constraint

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

注意

  • @NotNull 适用于任何类型被注解的元素必须不能与NULL

  • @NotEmpty 适用于String Map或者数组不能为Null且长度必须大于0

  • @NotBlank 只能用于String上面 不能为null,调用trim()后,长度必须大于0

使用

使用起来也非常简单,下面略过创建项目

模拟用户注册封装了一个UserDTO

当提交数据的时候如果使用以前的做法就是IF ELSE判断参数使用validator则是需要增加注解即可。

例如非空校验:

然后需要在controller方法体添加@Validated不加@Validated校验会不起作用

然后请求一下请求接口,把Email参数设置为空

参数:

{
"userName":"luomengsun",
"mobileNo":"11111111111",
"sex":1,
"age":21,
"email":""
}

返回结果:

后台抛出异常

这样是能校验成功,但是有个问题就是返回参数并不理想,前端也并不容易处理返回参数,所以我们添加一下全局异常处理,然后添加一下全局统一返回参数这样比较规范。

添加全局异常

创建一个GlobalExceptionHandler类,在类上方添加@RestControllerAdvice注解然后添加以下代码:

    /*** 方法参数校验*/
@ExceptionHandler(MethodArgumentNotValidException.class)
public ReturnVO handleMethodArgumentNotValidException(MethodArgumentNotValidException e) {log.error(e.getMessage(), e);return new ReturnVO().error(e.getBindingResult().getFieldError().getDefaultMessage());}

此方法主要捕捉MethodArgumentNotValidException异常然后对异常结果进行封装,如果需要在自行添加其他异常处理。

添加完之后我们在看一下运行结果,调用接口返回:

{"code": "9999","desc": "邮箱不能为空","data": null
}

OK 已经对异常进行处理。

校验格式

如果想要校验邮箱格式或者手机号的话也非常简单。

校验邮箱

    /*** 邮箱*/@NotBlank(message = "邮箱不能为空")@NotNull(message = "邮箱不能为空")@Email(message = "邮箱格式错误")private String email;

使用正则校验手机号

校验手机号使用正则进行校验,然后限制了一下位数

    /*** 手机号*/@NotNull(message = "手机号不能为空")@NotBlank(message = "手机号不能为空")@Pattern(regexp ="^[1][3,4,5,6,7,8,9][0-9]{9}$", message = "手机号格式有误")@Max(value = 11,message = "手机号只能为{max}位")@Min(value = 11,message = "手机号只能为{min}位")private String mobileNo;

查看一下运行结果

传入参数:

{
"userName":"luomengsun",
"mobileNo":"111111a",
"sex":1,
"age":21,
"email":"1212121"
}

返回结果:

{"code": "9999","desc": "邮箱格式错误","data": null
}

这里不再验证手机号的例子

自定义注解

上面的注解只有这么多,如果有特殊校验的参数我们可以使用Validator自定义注解进行校验

首先创建一个IdCard注解类

@Documented
@Target({ElementType.PARAMETER, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = IdCardValidator.class)
public @interface IdCard {String message() default "身份证号码不合法";Class<?>[] groups() default {};Class<? extends Payload>[] payload() default {};}

在UserDTO中添加@IdCard注解即可验证,在运行时触发,本文不对自定义注解做过多的解释,下篇文章介绍自定义注解

  • message 提示信息

  • groups 分组

  • payload  针对于Bean

然后添加IdCardValidator 主要进行验证逻辑

上面调用了is18ByteIdCardComplex方法,传入参数就是手机号,验证身份证规则自行百度:see_no_evil:

然后使用

    @NotNull(message = "身份证号不能为空")@IdCard(message = "身份证不合法")private String IdCardNumber;

分组

就比如上面我们定义的UserDTO中的参数如果要复用的话怎么办?

在重新定义一个类然后里面的参数要重新添加注解?

Validator提供了分组方法完美了解决DTO服用问题

现在我们注册的接口修改一下规则,只有用户名不能为空其他参数都不进行校验

先创建分组的接口

public interface Create  extends Default {
}

我们只需要在注解加入分组参数即可例如:

  /*** 用户名*/@NotBlank(message = "用户姓名不能为空",groups = Create.class)@NotNull(message = "用户姓名不能为空",groups = Create.class)private String userName;@NotBlank(message = "邮箱不能为空",groups = Update.class)@NotNull(message = "邮箱不能为空",groups = Update.class)@Email(message = "邮箱格式错误",groups = Update.class)private String email;

然后在修改Controller在@Validated中传入Create.class

    @PostMapping("/user")public ReturnVO userRegistra(@RequestBody @Validated(Create.class) UserDTO userDTO){ReturnVO returnVO = userService.userRegistra(userDTO);return returnVO ;}

然后调用传入参数:

{
"userName":"",
}

返回参数:

{"code": "9999","desc": "用户姓名不能为空","data": null
}

OK 现在只对Create的进行校验,而Updata组的不校验,如果需要复用DTO的话可以使用分组校验

校验单个参数

在开发的时候一定遇到过单个参数的情况,在参数前面加上注解即可

    @PostMapping("/get")public ReturnVO getUserInfo(@RequestParam("userId") @NotNull(message = "用户ID不能为空") String userId){return new ReturnVO().success();}

然后在Controller类上面增加@Validated注解,注意不是增加在参数前面。

本文通过OpenWrite的Markdown转换工具发布

关注我,回复“加群”加入各种主题讨论群

  • 图文结合!一文搞懂 Redis 常用知识点!

  • DD推荐:编辑器中的翻译神器!参数命名更轻松!

  • 传统网站性能优化的三种手段

  • Spring Boot Admin 2.2.0发布,新增中文展示!

  • IntelliJ IDEA 2019.3发布,2019.2 终成过去式

朕已阅 

Spring Boot如何优雅的校验参数相关推荐

  1. Spring Boot 如何优雅的校验参数?

    今天介绍一下 Spring Boot 如何优雅的整合JSR-303进行参数校验,说到参数校验可能都用过,但网上的教程大多是简单的介绍,所以我们今天详细看来一下 . 什么是 JSR-303? JSR-3 ...

  2. Spring Boot (16)---优雅的入门篇

    Spring Boot (16)---优雅的入门篇 Spring一直是很火的一个开源框架,在过去的一段时间里,Spring Boot在社区中热度一直很高,所以决定花时间来了解和学习,为自己做技术储备. ...

  3. Spring Boot 进行优雅的字段校验,写得太好了!

    作者 | 何甜甜在吗 来源 | juejin.cn/post/6913735652806754311 前段时间提交代码审核,同事提了一个代码规范缺陷:参数校验应该放在controller层.到底应该如 ...

  4. springMVC 优雅的校验参数(@Valid和@Validated)

    很痛苦遇到大量的参数进行校验,在业务中还要抛出异常或者不断的返回异常时的校验信息,在代码中相当冗长, 充满了if-else这种校验代码,今天我们就来学习spring的javax.validation ...

  5. Spring Boot 动态修改定时任务cron参数

    动态修改定时任务cron参数 不需要重启应用就可以动态的改变Cron表达式的值不能使用@Scheduled(cron = "${jobs.cron}")实现 动态定时任务类Dyna ...

  6. Spring Boot(Cloud) 优雅停机

    为了解决在微服务重启的过程中,可能出现一部分 http 请求处理失败的问题,提供一下方案 拟用方案: 第一步:重启前先从主动将服务剔除,并等待一段时间 第二步:停止服务并重启 一.主动将服务剔除 该方 ...

  7. 十、Spring boot 简单优雅的整合 Swagger2

    文章目录 前言 pom.xml SwaggerConfig 接口中的配置 番外 前言 swagger2 是什么,我这里就不说了,就是一个简单的接口文档,方便前后端联调. 其实之前没有想要到要使用swa ...

  8. Spring Boot通过ApplicationArguments获取args参数

    在具体使用Spring Boot的过程中,如果需要获得SpringApplication.run(args)方法传递的参数,那么可通过ApplicationArguments接口来获得. 使用方法非常 ...

  9. java护照号码校验_SpringBoot如何优雅的校验参数

    博客地址 :https://lqcoder.com 前言 做web开发有一点很烦人就是要校验参数,基本上每个接口都要对参数进行校验,比如一些格式校验 非空校验都是必不可少的.如果参数比较少的话还是容易 ...

最新文章

  1. c++重载(以运算符重载为主)
  2. Redis集群功能概述
  3. Tomcat NIO
  4. 有道云笔记到简书的迁移工具
  5. mysql innodb page_MySQL:Innodb page clean 线程 (一) 基础
  6. 修改pytho2安装插件的位置_office2016自定义安装以及修改安装位置
  7. WPF中TreeView.BringIntoView方法的替代方案
  8. leetcode329. 矩阵中的最长递增路径(dfs)
  9. java uuid 随机生成唯一序列号
  10. 查看oracle数据库版本
  11. 必备iOS设备解锁工具:iToolab UnlockGo for Mac(4.1.4中文)
  12. 苹果cms主动推送php,苹果cmsv10百度主动URL推送教程
  13. ValueError: The name None occurs multiple times, use a level number
  14. k8s 存活检查与就绪检查
  15. 短视频搬运如何上热门?搬运视频哪个平台容易挣钱?
  16. 【_declspec(dllimport)】_declspec(dllimport)
  17. 雷电2接口_雷电3和Type-C接口一样吗?差别很大
  18. Latex overleaf 图表公式参考文献
  19. 腾讯地图标记点击事件
  20. 图解 | 你管这破玩意儿叫TCP?

热门文章

  1. python3 实现 php bin2hex 函数
  2. linux shell expr命令 字符串操作
  3. linux c 结构体初始化的四种方法
  4. linux下使用binfmt_misc设定不同二进制的打开程序
  5. UNIX/LINUX程序设计教程(1)-- 获取系统信息
  6. C语言--对数组地址的解析
  7. 一个通用的任务管理模型-golang
  8. 新视野计算机等级考试官网,计算机二级C语言
  9. mysql file-pos_mysql-5.7 调整mysql的复制方式由master_log_file+master_log_pos 到gtid 详解
  10. oracle 12519,TNS-12519 与 processes 参数设置