JSR303校验
创始人
2024-05-30 12:14:17
0

1统一校验的需求

前端请求后端接口传输参数,是在controller中校验还是在Service中校验?

答案是都需要校验,只是分工不同。

Contoller中校验请求参数的合法性,包括:必填项校验,数据格式校验,比如:是否是符合一定的日期格式,等。

Service中要校验的是业务规则相关的内容,比如:课程已经审核通过所以提交失败。

Service中根据业务规则去校验不方便写成通用代码,Controller中则可以将校验的代码写成通用代码。

早在JavaEE6规范中就定义了参数校验的规范,它就是JSR-303,它定义了Bean Validation,即对bean属性进行校验。

SpringBoot提供了JSR-303的支持,它就是spring-boot-starter-validation,它的底层使用Hibernate Validator,Hibernate Validator是Bean Validation 的参考实现。

所以,我们准备在Controller层使用spring-boot-starter-validation完成对请求参数的基本合法性进行校验。

2统一校验实现

首先在Base工程添加spring-boot-starter-validation的依赖

org.springframework.bootspring-boot-starter-validation

现在准备对内容管理模块添加课程接口进行参数校验,如下接口

@ApiOperation("新增课程基础信息")
@PostMapping("/course")
public CourseBaseInfoDto createCourseBase(@RequestBody AddCourseDto addCourseDto){//机构id,由于认证系统没有上线暂时硬编码Long companyId = 1L;return courseBaseInfoService.createCourseBase(companyId,addCourseDto);
}

此接口使用AddCourseDto模型对象接收参数,所以进入AddCourseDto类,在属性上添加校验规则。

package com.xuecheng.content.model.dto;import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;import javax.validation.constraints.NotEmpty;
import javax.validation.constraints.Size;
import java.math.BigDecimal;/*** @description 添加课程dto* @author Mr.M* @date 2022/9/7 17:40* @version 1.0*/
@Data
@ApiModel(value="AddCourseDto", description="新增课程基本信息")
public class AddCourseDto {@NotEmpty(message = "课程名称不能为空")@ApiModelProperty(value = "课程名称", required = true)private String name;@NotEmpty(message = "适用人群不能为空")@Size(message = "适用人群内容过少",min = 10)@ApiModelProperty(value = "适用人群", required = true)private String users;@ApiModelProperty(value = "课程标签")private String tags;@NotEmpty(message = "课程分类不能为空")@ApiModelProperty(value = "大分类", required = true)private String mt;@NotEmpty(message = "课程分类不能为空")@ApiModelProperty(value = "小分类", required = true)private String st;@NotEmpty(message = "课程等级不能为空")@ApiModelProperty(value = "课程等级", required = true)private String grade;@ApiModelProperty(value = "教学模式(普通,录播,直播等)", required = true)private String teachmode;@ApiModelProperty(value = "课程介绍")private String description;@ApiModelProperty(value = "课程图片", required = true)private String pic;@NotEmpty(message = "收费规则不能为空")@ApiModelProperty(value = "收费规则,对应数据字典", required = true)private String charge;@ApiModelProperty(value = "价格")private BigDecimal price;}

上边用到了@NotEmpty和@Size两个注解,@NotEmpty表示属性不能为空,@Size表示限制属性内容的长短。

在javax.validation.constraints包下有很多这样的校验注解

在这里插入图片描述
规则如下:
在这里插入图片描述

定义好校验规则还需要开启校验,在controller方法中添加@Validated注解,如下:

@ApiOperation("新增课程基础信息")
@PostMapping("/course")
public CourseBaseInfoDto createCourseBase(@RequestBody @Validated AddCourseDto addCourseDto){//机构id,由于认证系统没有上线暂时硬编码Long companyId = 1L;return courseBaseInfoService.createCourseBase(companyId,addCourseDto);
}

如果校验出错Spring会抛出MethodArgumentNotValidException异常,我们需要在统一异常处理器中捕获异常,解析出异常信息。

代码 如下:

@ResponseBody
@ExceptionHandler(value = MethodArgumentNotValidException.class)
@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
public RestErrorResponse doValidException(MethodArgumentNotValidException argumentNotValidException) {BindingResult bindingResult = argumentNotValidException.getBindingResult();StringBuffer errMsg = new StringBuffer();List fieldErrors = bindingResult.getFieldErrors();fieldErrors.forEach(error -> {errMsg.append(error.getDefaultMessage()).append(",");});log.error(errMsg.toString());return new RestErrorResponse(errMsg.toString());
}

重启内容管理服务。

使用httpclient进行测试,将必填项设置为空,“适用人群” 属性的内容设置1个字。

执行测试,接口响应结果如下:

{"errMessage": "课程名称不能为空 课程分类不能为空 课程分类不能为空 适用人群内容过少 "
}

可以看到校验器生效。

3分组校验

有时候在同一个属性上设置一个校验规则不能满足要求,比如:订单编号由系统生成,在添加订单时要求订单编号为空,在更新 订单时要求订单编写不能为空。此时就用到了分组校验,同一个属性定义多个校验规则属于不同的分组,比如:添加订单定义@NULL规则属于insert分组,更新订单定义@NotEmpty规则属于update分组,insert和update是分组的名称,是可以修改的。

下边举例说明

我们用class类型来表示不同的分组,所以我们定义不同的接口类型(空接口)表示不同的分组,由于校验分组是公用的,所以定义在 base工程中。如下:

package com.xuecheng.base.execption;/*** @description 校验分组* @author Mr.M* @date 2022/9/8 15:05* @version 1.0*/
public class ValidationGroups {public interface Inster{};public interface Update{};public interface Delete{};}

下边在定义校验规则时指定分组:

@NotEmpty(groups = {ValidationGroups.Inster.class},message = "添加课程名称不能为空")@NotEmpty(groups = {ValidationGroups.Update.class},message = "修改课程名称不能为空")
// @NotEmpty(message = "课程名称不能为空")@ApiModelProperty(value = "课程名称", required = true)private String name;

在Controller方法中启动校验规则指定要使用的分组名:

@ApiOperation("新增课程基础信息")
@PostMapping("/course")
public CourseBaseInfoDto createCourseBase(@RequestBody @Validated({ValidationGroups.Inster.class}) AddCourseDto addCourseDto){//机构id,由于认证系统没有上线暂时硬编码Long companyId = 1L;return courseBaseInfoService.createCourseBase(companyId,addCourseDto);
}

再次测试,由于这里指定了Insert分组,所以抛出 异常信息:添加课程名称不能为空。

如果修改分组为ValidationGroups.Update.class,异常信息为:修改课程名称不能为空。

4校验规则不满足?

如果javax.validation.constraints包下的校验规则满足不了需求怎么办?

1、手写校验代码 。

2、自定义校验规则注解。

如何自定义校验规则注解,请自行查阅资料实现。

5面试

1、请求参数的合法性校验如何做?

使用基于JSR303的校验框架实现,SpringBoot提供了JSR-303的支持,它就是spring-boot-starter-validation,它包括了很多校验规则,只需要在模型类中通过注解指定校验规则,在controller方法上开启校验。

相关内容

热门资讯

我的同桌中英文作文【精选6篇... 我的同桌中英文作文 篇一Title: My Desk MateI have had the same...
清洁工的英语作文(精简6篇) 清洁工的英语作文 篇一The Importance of Cleaners in Our Socie...
又是一天作文200字(精选3... 又是一天作文200字 篇一阳光洒在我的脸上,唤醒了我美好的一天。我迅速起床,洗漱完毕后,走进厨房准备...
唯美英语小句子【经典6篇】 唯美英语小句子 篇一In the fast-paced world we live in, it i...
自我优秀介绍范文英语作文60... 自我优秀介绍范文英语作文 第一篇My name’s Lin Yanyi, I’m eight, I’...
英语解析与检测作文范文(精简... 英语解析与检测作文范文 篇一标题:The Importance of Learning Englis...
高考英语劳动教育范文7篇(精... 高考英语劳动教育范文7篇 篇一:劳动教育的重要性劳动教育是培养学生实践能力和劳动意识的重要途径。在过...
我的学校英语作文 我的学校英语作文(精选21篇)  在日常学习、工作或生活中,大家对作文都再熟悉不过了吧,写作文可以锻...
This is me英语作文 This is me英语作文2篇  在日复一日的学习、工作或生活中,大家都不可避免地会接触到作文吧,...
他伤心了作文范文英语【优秀6... 他伤心了作文范文英语 篇一Title: The Heartache of His SadnessIn...
成人高考英语作文【精彩6篇】 成人高考英语作文 篇一The Importance of Lifelong LearningIn t...
旅行计划英语作文【优秀6篇】 旅行计划英语作文 篇一My Dream Trip to ParisI have always dre...
我的英语老师英语作文(优质6... 我的英语老师英语作文 篇一我有一个非常出色的英语老师,她是我最喜欢的老师之一。她不仅教授我们英语知识...
关于交通的英语句子【通用3篇... 关于交通的英语句子 篇一Title: The Importance of Traffic Educa...
大一英语作文简介范文23篇(... 大一英语作文简介范文23篇 篇一标题:The Importance of Time Manageme...
自我介绍英语作文 自我介绍英语作文通用10篇  在日常学习、工作或生活中,大家都跟作文打过交道吧,通过作文可以把我们那...
写我的朋友的英语作文400字... 篇一:我的朋友My Friend篇一:我的朋友My FriendMy friend's name i...
英语作文书写优秀范文(精简6... 英语作文书写优秀范文 篇一Title: The Importance of Learning Eng...
小升初英语作文(优选3篇) 小升初英语作文 篇一My Dream JobEveryone has a dream job in ...
沪教版小学英语范文(优质6篇... 沪教版小学英语范文 篇一:My FamilyHello, everyone! My name is ...