乐清建网站,建网站要多长时间,请人做软件开发的网站,手机网站建站用哪个软件好前言在日常的开发中#xff0c;参数校验是非常重要的一个环节#xff0c;严格参数校验会减少很多出bug的概率#xff0c;增加接口的安全性。在此之前写过一篇SpringBoot统一参数校验主要介绍了一些简单的校验方法。而这篇则是介绍一些进阶的校验方式。比如说#xff1a;在某… 前言 在日常的开发中参数校验是非常重要的一个环节严格参数校验会减少很多出bug的概率增加接口的安全性。在此之前写过一篇SpringBoot统一参数校验主要介绍了一些简单的校验方法。而这篇则是介绍一些进阶的校验方式。比如说在某个接口编写的过程中肯定会遇到当xxType值为AparamA值必传。xxType值为BparamB值必须传。对于这样的通常的做法就是在controller加上各种if判断。显然这样的代码是不够优雅的而分组校验及自定义参数校验就是来解决这个问题的。PathVariable参数校验 Restful的接口在现在来讲应该是比较常见的了常用的地址栏的参数我们都是这样校验的。/** * 获取电话号码信息 */GetMapping(/phoneInfo/{phone})public ResultVo phoneInfo(PathVariable(phone) String phone){ // 验证电话号码是否有效 String pattern ^[1][3,4,5,7,8][0-9]{9}$; boolean isValid Pattern.matches(pattern, phone); if(isValid){ // 执行相应逻辑 return ResultVoUtil.success(phone); } else { // 返回错误信息 return ResultVoUtil.error(手机号码无效); }}很显然上面的代码不够优雅所以我们可以在参数后面添加对应的正则表达式phone:正则表达式来进行验证。这样就省去了在controller编写校验代码了。/** * 获取电话号码信息 */GetMapping(/phoneInfo/{phone:^[1][3,4,5,7,8][0-9]{9}$})public ResultVo phoneInfo(PathVariable(phone) String phone){ return ResultVoUtil.success(phone);}虽然这样处理后代码更精简了。但是如果传入的手机号码不符合规则会直接返回404。而不是提示手机号码错误。错误信息如下自定义校验注解 我们以校验手机号码为例虽然validation提供了Pattern这个注解来使用正则表达式进行校验。如果被使用在多处一旦正则表达式发生更改则需要一个一个的进行修改。很显然为了避免做这样的无用功自定义校验注解就是你的好帮手。Datapublic class PhoneForm { /** * 电话号码 */ Pattern(regexp ^[1][3,4,5,7,8][0-9]{9}$ , message 电话号码有误) private String phone;} 要实现一个自定义校验注解主要是有两步。一是注解本身二是校验逻辑实现类。PhoneVerify 校验注解Target({ElementType.FIELD})Retention(RetentionPolicy.RUNTIME)Constraint(validatedBy PhoneValidator.class)public interface Phone { String message() default 手机号码格式有误; Class[] groups() default {}; Class extends Payload[] payload() default {};}PhoneValidator 校验实现类public class PhoneValidator implements ConstraintValidatorPhone, Object { Override public boolean isValid(Object telephone, ConstraintValidatorContext constraintValidatorContext) { String pattern ^1[3|4|5|7|8]\\d{9}$; return Pattern.matches(pattern, telephone.toString()); }}CustomForm 表单数据Datapublic class CustomForm { /** * 电话号码 */ Phone private String phone;}测试接口PostMapping(/customTest)public ResultVo customTest(RequestBody Validated CustomForm form){ return ResultVoUtil.success(form.getPhone());}注解的含义Target({ElementType.FIELD}) 注解是指定当前自定义注解可以使用在哪些地方这里仅仅让他可以使用属性上。但还可以使用在更多的地方比如说方法上、构造器上等等。TYPE - 类接口(包括注解类型)或枚举FIELD - 字段(包括枚举常量)METHOD - 方法PARAMETER - 参数CONSTRUCTOR - 构造函数LOCAL_VARIABLE - 局部变量ANNOTATION_TYPE -注解类型PACKAGE - 包TYPE_PARAMETER - 类型参数TYPE_USE - 使用类型Retention(RetentionPolicy.RUNTIME) 指定当前注解保留到运行时。保留策略有下面三种SOURCE - 注解只保留在源文件当Java文件编译成class文件的时候注解被遗弃。CLASS - 注解被保留到class文件但jvm加载class文件时候被遗弃这是默认的生命周期。RUNTIME - 注解不仅被保存到class文件中jvm加载class文件之后仍然存在。Constraint(validatedBy PhoneValidator.class) 指定了当前注解使用哪个校验类来进行校验。分组校验UserFormDatapublic class UserForm { /** * id */ Null(message 新增时id必须为空, groups {Insert.class}) NotNull(message 更新时id不能为空, groups {Update.class}) private String id; /** * 类型 */ NotEmpty(message 姓名不能为空 , groups {Insert.class}) private String name; /** * 年龄 */ NotEmpty(message 年龄不能为空 , groups {Insert.class}) private String age; }Insert分组public interface Insert {}Update分组public interface Update {}测试接口/** * 添加用户 */PostMapping(/addUser)public ResultVo addUser(RequestBody Validated({Insert.class}) UserForm form){ // 选择对应的分组进行校验 return ResultVoUtil.success(form);}/** * 更新用户 */PostMapping(/updateUser)public ResultVo updateUser(RequestBody Validated({Update.class}) UserForm form){ // 选择对应的分组进行校验 return ResultVoUtil.success(form);}测试结果添加测试更新测试顺序校验GroupSequence 在GroupSequence内可以指定分组校验的顺序。比如说GroupSequence({Insert.class, Update.class, UserForm.class})先执行Insert校验然后执行Update校验。如果Insert分组校验失败了则不会进行Update分组的校验。DataGroupSequence({Insert.class, Update.class, UserForm.class})public class UserForm { /** * id */ Null(message 新增时id必须为空, groups {Insert.class}) NotNull(message 更新时id不能为空, groups {Update.class}) private String id; /** * 类型 */ NotEmpty(message 姓名不能为空 , groups {Insert.class}) private String name; /** * 年龄 */ NotEmpty(message 年龄不能为空 , groups {Insert.class}) private String age;}测试接口/*** 编辑用户*/PostMapping(/editUser)public ResultVo editUser(RequestBody Validated UserForm form){ return ResultVoUtil.success(form);}测试结果 哈哈哈测试结果其实是个死循环不管你咋输入都会报错小伙伴可以尝试一下哦。上面的例子只是个演示在实际中还是别这样做了需要根据具体逻辑进行校验。自定义分组校验 对于之前提到了当xxType值为AparamA值必传。xxType值为BparamB值必须传这样的场景。单独使用分组校验和分组序列是无法实现的。需要使用GroupSequenceProvider才行。自定义分组表单DataGroupSequenceProvider(value CustomSequenceProvider.class)public class CustomGroupForm { /** * 类型 */ Pattern(regexp [A|B] , message 类型不必须为 A|B) private String type; /** * 参数A */ NotEmpty(message 参数A不能为空 , groups {WhenTypeIsA.class}) private String paramA; /** * 参数B */ NotEmpty(message 参数B不能为空, groups {WhenTypeIsB.class}) private String paramB; /** * 分组A */ public interface WhenTypeIsA { } /** * 分组B */ public interface WhenTypeIsB { }}CustomSequenceProviderpublic class CustomSequenceProvider implements DefaultGroupSequenceProviderCustomGroupForm { Override public List getValidationGroups(CustomGroupForm form) { List defaultGroupSequence new ArrayList(); defaultGroupSequence.add(CustomGroupForm.class);if (form ! null A.equals(form.getType())) { defaultGroupSequence.add(CustomGroupForm.WhenTypeIsA.class); }if (form ! null B.equals(form.getType())) { defaultGroupSequence.add(CustomGroupForm.WhenTypeIsB.class); }return defaultGroupSequence; }}测试接口/** * 自定义分组 */PostMapping(/customGroup)public ResultVo customGroup(RequestBody Validated CustomGroupForm form){ return ResultVoUtil.success(form);}测试结果Type类型为AType类型为B小结一下 GroupSequence注解是一个标准的Bean认证注解。正如之前它能够让你静态的重新定义一个类的默认校验组顺序。然而GroupSequenceProvider它能够让你动态的定义一个校验组的顺序。注意的一个点SpringBoot 2.3.x 移除了validation依赖需要手动引入依赖。dependency groupIdorg.springframework.bootgroupId artifactIdspring-boot-starter-validationartifactIddependency总结 个人的一些小经验参数的非空判断这个应该是校验的第一步了除了非空校验我们还需要做到下面这几点普通参数 - 需要限定字段的长度。如果会将数据存入数据库长度以数据库为准反之根据业务确定。类型参数 - 最好使用正则对可能出现的类型做到严格校验。比如type的值是【0|1|2】这样的。列表(list)参数 - 不仅需要对list内的参数是否合格进行校验还需要对list的size进行限制。比如说 100。日期邮件金额URL这类参数都需要使用对于的正则进行校验。参数真实性 - 这个主要针对于 各种Id 比如说 userId、merchantId对于这样的参数都需要进行真实性校验判断系统内是有含有并且对应的状态是否正常。 参数校验越严格越好严格的校验规则不仅能减少接口出错的概率同时还能避免出现脏数据从而来保证系统的安全性和稳定性。错误的提醒信息需要友好一点哦防止等下被前端大哥吐槽哦。上期回顾SpringBoot统一参数校验结尾 如果觉得对你有帮助可以多多评论多多点赞哦也可以到我的主页看看说不定有你喜欢的文章也可以随手点个关注哦谢谢。 我是不一样的科技宅每天进步一点点体验不一样的生活。我们下期见