如何使用JSR 303 进行后台数据校验?
创始人
2024-05-29 23:10:17
0

文章目录

  • 一、JSR 303
    • 1、什么是 JSR 303?
    • 2、为什么使用 JSR 303?
    • 3、JSR 303 常见操作?
    • 二、使用 JSR 303 相关注解处理逻辑
    • 1、JSR 303 注解处理逻辑
      • 1.1 使用步骤
      • 1.2 实际应用
    • 2 、JSR 303 分组校验
      • 2.1 为什么使用 分组校验?
      • 2.2 使用步骤
      • 2.3 实际应用
    • 3、JSR 303 自定义校验注解
      • 3.1 为什么使用自定义校验注解?
      • 3.2 使用步骤
      • 3.3 实际应用

三连哦

一、JSR 303

1、什么是 JSR 303?

  • JSR 是 Java Specification Requests 的缩写,即 Java 规范提案。
  • 存在各种各样的 JSR,简单的理解为 JSR 是一种 Java 标准。
  • JSR 303 就是数据检验的一个标准(Bean Validation (JSR 303))。
参考:https://www.jianshu.com/p/554533f88370

2、为什么使用 JSR 303?

  • 处理一段业务逻辑,首先要确保数据输入的正确性,所以需要先对数据进行检查,保证数据在语义上的正确性,再根据数据进行下一步的处理。
  • 前端可以通过 js 程序校验数据是否合法,后端同样也需要进行校验。而后端最简单的实现就是直接在业务方法中对数据进行处理,但是不同的业务方法可能会出现同样的校验操作,这样就出现了数据的冗余。为了解决这个情况,JSR 303 出现了。
  • JSR 303 使用 Bean Validation,即在 Bean 上添加相应的注解,去实现数据校验。这样在执行业务方法前,都会根据注解对数据进行校验,从而减少自定义的校验逻辑,减少代码冗余。

3、JSR 303 常见操作?

(1)可以通过简单的注解校验 Bean 属性,比如 @NotNull、@Null 等。
(2)可以通过 Group 分组自定义需要执行校验的属性。
(3)可以自定义注解并指定校验规则。
(4)支持基于 JSR 303 的实现,比如 Hibernate Validator(额外添加一些注解)。

二、使用 JSR 303 相关注解处理逻辑

1、JSR 303 注解处理逻辑

1.1 使用步骤

  • 1、在相关的 Bean 上标注需要处理的注解,并指定需要提示的信息(若不指定,会从默认配置文件中读取默认的信息)。

  • 2、在相关的方法上,使用 @Valid 注解(或者 @Validated 指定组名)标记需要被校验的数据,否则会不生效。

注意:检测到数据异常后,系统会向外抛出异常,如果做了统一异常处理,可以根据 postman 测试的结果,找到控制台打印出的相应的异常,并处理。

  • 3、处理异常。使用 BindingResult 可以获取到检测结果,然后进行处理。也可以使用 全局统一异常 处理(@RestControllerAdvice@ExceptionHandler),处理检测结果

1.2 实际应用

1、给Bean添加校验注解:javax.validation.constraints,并定义自己的message提示


import javax.validation.constraints.*;/*** 品牌* * @author zhengyuzhu* @email 2977429967@qq.com* @date 2023-02-25 23:11:31*/
@Data
@TableName("pms_brand")
public class BrandEntity implements Serializable {private static final long serialVersionUID = 1L;/*** 品牌id*/@NotNull(message = "修改必须指定品牌id")@TableIdprivate Long brandId;/*** 品牌名*/@NotBlank(message = "品牌名必须提交")private String name;}

2、修改 Controller 方法,使用 @Valid 注解标记需要检测的数据、比如在常见的增加和修改方法上,这个时候需要将前台传来的数据插入数据库


@RestController
@RequestMapping("product/brand")
public class BrandController {@Autowiredprivate BrandService brandService;/*** 保存新添加品牌*/@RequestMapping("/save")public R save(@Validated @RequestBody BrandEntity brand){brandService.save(brand);return R.ok();}/*** 修改*/@RequestMapping("/update")public R update(@Validated @RequestBody BrandEntity brand){brandService.updateDetail(brand);return R.ok();}}

提示:这个时候如果插入的数据不符合规范,就会抛出异常、为了避免代码冗余。可以使用全局异常来进行处理

3、集中处理异常

可以使用 BindingResult 去处理捕获到的数据并进行相关处理

/*** 集中处理所有异常*/
@Slf4j@RestControllerAdvice(basePackages = "com.zyz.gulimall.product.controller")
public class GulimallExceptionControllerAdvice {@ExceptionHandler(value= MethodArgumentNotValidException.class)public R handleVaildException(MethodArgumentNotValidException e){log.error("数据校验出现问题{},异常类型:{}",e.getMessage(),e.getClass());BindingResult bindingResult = e.getBindingResult();Map errorMap = new HashMap<>();// 获取校验结果,遍历获取捕获到的每个校验结果bindingResult.getFieldErrors().forEach((fieldError)->{// 存储得到的校验结果errorMap.put(fieldError.getField(),fieldError.getDefaultMessage());});return R.error(BizCodeEnume.VAILD_EXCEPTION.getCode(),BizCodeEnume.VAILD_EXCEPTION.getMsg()).put("data",errorMap);}@ExceptionHandler(value = Throwable.class)public R handleException(Throwable throwable){log.error("错误:",throwable);return R.error(BizCodeEnume.UNKNOW_EXCEPTION.getCode(),BizCodeEnume.UNKNOW_EXCEPTION.getMsg());}}

2 、JSR 303 分组校验

2.1 为什么使用 分组校验?

如果出现多个方法,都需要校验 Bean,且校验规则不同的时候,怎么办呢?

就比如下方,修改品牌ID的时候,提交为空的时候,提示用户的应该是修改ID不能为空。新增的时候,应该是提示用户,新增ID不能为空。但是你也可以直接写ID不能为空,但总感觉缺少点啥。

分组校验就可以去解决该问题,每个分组指定不同的校验规则,不同的方法执行不同的分组,就可以得到不同的校验结果。 也就是说,定义好分组后,在校验的时候,就可以去找对应的分组进行相关的信息提示。

	/*** 品牌id*/@NotNull(message = "修改必须指定品牌id",groups = {UpdateGroup.class})@Null(message = "新增不能指定id",groups = {AddGroup.class})@TableIdprivate Long brandId;

2.2 使用步骤

  • 1、定义一个空接口,用于指定分组,内部不需要任何实现。

  • 2、指定 注解时,通过 groups 指定分组。用于指定在某个分组条件下,才去执行校验规则。

  • 3、在相关的业务方法上,通过 @Validated 注解指定分组,去指定校验。

注:使用分组校验后,Bean 注解上若不指定分组,则不会执行校验规则。

2.3 实际应用

1 创建两个新增和修改接口

public interface AddGroup {
}public interface UpdateGroup {
}

2 指定 注解时,通过 groups 指定分组

比如,这里的品牌id定义两个分组,一个是增加时候的校验,一个是修改时候的校验,对应不同的校验规则。也可以将通用的校验规则,以分组的时候同时管理多个。

	/*** 品牌id*/@NotNull(message = "修改必须指定品牌id",groups = {UpdateGroup.class})@Null(message = "新增不能指定id",groups = {AddGroup.class})@TableIdprivate Long brandId;/*** 品牌名*/@NotBlank(message = "品牌名必须提交",groups = {AddGroup.class,UpdateGroup.class})private String name;

3、在相关的业务方法上,通过 @Validated 注解指定分组,去指定校验。

    /*** 保存新添加品牌*/@RequestMapping("/save")public R save(@Validated({AddGroup.class}) @RequestBody BrandEntity brand){brandService.save(brand);return R.ok();}/*** 修改*/@RequestMapping("/update")public R update(@Validated(UpdateGroup.class) @RequestBody BrandEntity brand){brandService.updateDetail(brand);return R.ok();}

在这里插入图片描述

3、JSR 303 自定义校验注解

3.1 为什么使用自定义校验注解?

当上面的注解满足不了业务需求时,可以自定义校验注解,自定义校验规则。

3.2 使用步骤

  • 1、需要自定义一个校验注解。可以创建一个 ValidationMessages.properties 用于保存默认的 message 信息。

  • 2、需要自定义一个校验器,即自定义校验规则。实现 ConstraintValidator 接口,并重写相关方法。

注:initialize 方法用于初始化,可以获取 自定义的属性的值。isValid
方法用于校验,可以获取到实际的值,然后与自定义的属性值进行比较。

  • 3、将校验注解 与 校验器 关联起来。@Constraint(validatedBy = {TestValidConstraintValidator.class})

3.3 实际应用

如下例,自定义一个校验规则,判断数据是否是 0,1。当传来的数据不是这两种之一,校验不通过。

  • 1、自定义一个校验注解
/*** @author zyz*/
@Documented
@Constraint(validatedBy = { ListValueConstraintValidator.class })
@Target({ METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER, TYPE_USE })
@Retention(RUNTIME)
public @interface ListValue {/*** 添加校验* @return*/String message() default "{com.zyz.common.valid.ListValue.message}";Class[] groups() default { };Class[] payload() default { };int[] vals() default { };
}

创建一个文件ValidationMessages.properties。当不符合自定义规则校验的时候,message可以获取到对应的信息

配置文件内容:com.zyz.common.valid.ListValue.message=必须提交指定的值啊

在这里插入图片描述

  • 2 自定义一个校验器TestValidConstraintValidator, 用于检测值是否合法。
/*** @author zyz*//*** 实现 ConstraintValidator 接口,* 其中 ConstraintValidator 的泛型,一个需要指定自定义的注解,一个需要指定需要获取的值的类型。* 比如:*  ConstraintValidator 中*      ListValue   表示自定义注解*      String      表示获取的值的类型* 即定义规则,判断一个 String 的值的长度是否满足条件*/
public class ListValueConstraintValidator implements ConstraintValidator {private Set set = new HashSet<>();/*** 初始化方法*/@Overridepublic void initialize(ListValue constraintAnnotation) {int[] vals = constraintAnnotation.vals();for (int val : vals) {set.add(val);}}//判断是否校验成功/**** @param value 需要校验的值* @param context* @return*/@Overridepublic boolean isValid(Integer value, ConstraintValidatorContext context) {return set.contains(value);}
}
  • 3、使用注解

使用自定义注解,这里的@ListValue 就是使用的自定义注解,当前端传来状态的值不是对应数据。就会走自定义校验判断、提示对应的信息。这里也使用了分组的形式。

	/*** 显示状态[0-不显示;1-显示]*/@NotNull(groups = {AddGroup.class, UpdateStatusGroup.class})@ListValue(vals={0,1},groups = {AddGroup.class, UpdateStatusGroup.class})private Integer showStatus;

方法中使用

这里使用了分组,就会自动校验

    /*** 保存新添加品牌*/@RequestMapping("/save")public R save(@Validated({AddGroup.class}) @RequestBody BrandEntity brand){brandService.save(brand);return R.ok();}

资料参考:JSR 303

相关内容

热门资讯

DNA解码者徐海波 圆200余... 转自:邯郸晚报 “任何一个犯罪现场,作案人必定会留下痕迹,刑事技术人员的工作就是让这些物证‘开口说话...
曲周公安紧急护送儿童就医 转自:邯郸晚报 近日,曲周县公安局白寨派出所接到辖区群众报警,称其孩子误食灭虫的药物,急需前往医院救...
英可瑞跌2.04%,成交额29... 5月14日,英可瑞盘中下跌2.04%,截至10:08,报16.33元/股,成交2912.85万元,换...
中方敦促:立即停止军事进攻 转自:扬子晚报中方呼吁采取紧急行动为加沙人道惨剧画上句号联合国安理会13日举行加沙人道局势紧急公开会...
国际金价回调逼近3200美元/... 证券时报记者 赵黎昀国际政经形势多变的当下,贵金属市场波动也牵动着市场神经。“五一”小长假过后,国际...
海信家电跌2.02%,成交额6... 5月14日,海信家电盘中下跌2.02%,截至10:07,报29.08元/股,成交6009.03万元,...
恒光股份涨2.47%,成交额3... 5月14日,恒光股份盘中上涨2.47%,截至10:08,报21.61元/股,成交3376.66万元,...
张姓股民向*ST美谷发起索赔 ...   受损股民可至Hehson股民维权平台登记该公司维权:http://wq.finance.sina...
让世界充满爱,中国太保发布 “... 转自:上观新闻5月13日,在迎来建司34周年和第4个“中国太保公益日”之际,上海儿童艺术剧场内一场充...
富岭股份跌2.00%,成交额1... 5月14日,富岭股份盘中下跌2.00%,截至10:08,报15.65元/股,成交1.97亿元,换手率...
久之洋跌2.03%,成交额32... 5月14日,久之洋盘中下跌2.03%,截至10:07,报32.85元/股,成交3229.30万元,换...
中恒电气跌2.01%,成交额2... 5月14日,中恒电气(维权)盘中下跌2.01%,截至10:07,报16.05元/股,成交2.07亿元...
鸿泉物联涨2.01%,成交额2... 5月14日,鸿泉物联盘中上涨2.01%,截至10:00,报32.47元/股,成交2063.34万元,...
欧圣电气跌2.30%,成交额7... 5月14日,欧圣电气盘中下跌2.30%,截至10:07,报36.50元/股,成交7695.58万元,...
长沙代孕机构窝点被端18人落网... 【#长沙代孕机构窝点被端18人落网# 幕后负责人及两名医护被控制】#全力追击长沙代孕机构其他涉案人员...
信隆健康涨2.08%,成交额4... 5月14日,信隆健康盘中上涨2.08%,截至10:07,报7.86元/股,成交4034.89万元,换...
医疗器械行业遇冷,IVD板块承...   炒股就看金麒麟分析师研报,权威,专业,及时,全面,助您挖掘潜力主题机会!         《电...
联得装备跌2.01%,成交额3... 5月14日,联得装备盘中下跌2.01%,截至09:50,报30.24元/股,成交3098.98万元,...
永杉锂业涨2.64%,成交额2... 5月14日,永杉锂业盘中上涨2.64%,截至10:02,报8.93元/股,成交2538.13万元,换...
韩国空难遇难者家属提起刑事诉讼 来源:参考消息网参考消息网5月14日报道据新加坡《联合早报》网站5月13日报道,韩国济州航空客机空难...