springboot整合JSR303校驗(yàn)功能實(shí)現(xiàn)代碼
JSR303簡(jiǎn)介
JSR-303 是 JAVA EE 6 中的一項(xiàng)子規(guī)范,叫做 Bean Validation,官方參考實(shí)現(xiàn)是Hibernate Validator。Hibernate Validator 提供了 JSR 303 規(guī)范中所有內(nèi)置 constraint 的實(shí)現(xiàn),除此之外還有一些附加的 constraint。
4.7 JSR303校驗(yàn)
4.7.1 統(tǒng)一校驗(yàn)的需求
前端請(qǐng)求后端接口傳輸參數(shù),是在controller中校驗(yàn)還是在Service中校驗(yàn)?
答案是都需要校驗(yàn),只是分工不同。
Contoller中校驗(yàn)請(qǐng)求參數(shù)的合法性,包括:必填項(xiàng)校驗(yàn),數(shù)據(jù)格式校驗(yàn),比如:是否是符合一定的日期格式,等。
Service中要校驗(yàn)的是業(yè)務(wù)規(guī)則相關(guān)的內(nèi)容,比如:課程已經(jīng)審核通過(guò)所以提交失敗。
Service中根據(jù)業(yè)務(wù)規(guī)則去校驗(yàn)不方便寫成通用代碼,Controller中則可以將校驗(yàn)的代碼寫成通用代碼。
早在JavaEE6規(guī)范中就定義了參數(shù)校驗(yàn)的規(guī)范,它就是JSR-303,它定義了Bean Validation,即對(duì)bean屬性進(jìn)行校驗(yàn)。
SpringBoot提供了JSR-303的支持,它就是spring-boot-starter-validation,它的底層使用Hibernate Validator,Hibernate Validator是Bean Validation 的參考實(shí)現(xiàn)。
所以,我們準(zhǔn)備在Controller層使用spring-boot-starter-validation完成對(duì)請(qǐng)求參數(shù)的基本合法性進(jìn)行校驗(yàn)。
4.7.2 統(tǒng)一校驗(yàn)實(shí)現(xiàn)
首先在Base工程添加spring-boot-starter-validation的依賴
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-validation</artifactId>
</dependency>現(xiàn)在準(zhǔn)備對(duì)內(nèi)容管理模塊添加課程接口進(jìn)行參數(shù)校驗(yàn),如下接口
@ApiOperation("新增課程基礎(chǔ)信息")
@PostMapping("/course")
public CourseBaseInfoDto createCourseBase(@RequestBody AddCourseDto addCourseDto){
//機(jī)構(gòu)id,由于認(rèn)證系統(tǒng)沒(méi)有上線暫時(shí)硬編碼
Long companyId = 1L;
return courseBaseInfoService.createCourseBase(companyId,addCourseDto);
}此接口使用AddCourseDto模型對(duì)象接收參數(shù),所以進(jìn)入AddCourseDto類,在屬性上添加校驗(yàn)規(guī)則。
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 = "適用人群內(nèi)容過(guò)少",min = 10)
@ApiModelProperty(value = "適用人群", required = true)
private String users;
@ApiModelProperty(value = "課程標(biāo)簽")
private String tags;
@NotEmpty(message = "課程分類不能為空")
@ApiModelProperty(value = "大分類", required = true)
private String mt;
@NotEmpty(message = "課程分類不能為空")
@ApiModelProperty(value = "小分類", required = true)
private String st;
@NotEmpty(message = "課程等級(jí)不能為空")
@ApiModelProperty(value = "課程等級(jí)", required = true)
private String grade;
@ApiModelProperty(value = "教學(xué)模式(普通,錄播,直播等)", required = true)
private String teachmode;
@ApiModelProperty(value = "課程介紹")
private String description;
@ApiModelProperty(value = "課程圖片", required = true)
private String pic;
@NotEmpty(message = "收費(fèi)規(guī)則不能為空")
@ApiModelProperty(value = "收費(fèi)規(guī)則,對(duì)應(yīng)數(shù)據(jù)字典", required = true)
private String charge;
@ApiModelProperty(value = "價(jià)格")
private BigDecimal price;
}
上邊用到了@NotEmpty和@Size兩個(gè)注解,@NotEmpty表示屬性不能為空,@Size表示限制屬性內(nèi)容的長(zhǎng)短。
在javax.validation.constraints包下有很多這樣的校驗(yàn)注解

規(guī)則如下:

定義好校驗(yàn)規(guī)則還需要開(kāi)啟校驗(yàn),在controller方法中添加@Validated注解,如下:
@ApiOperation("新增課程基礎(chǔ)信息")
@PostMapping("/course")
public CourseBaseInfoDto createCourseBase(@RequestBody @Validated AddCourseDto addCourseDto){
//機(jī)構(gòu)id,由于認(rèn)證系統(tǒng)沒(méi)有上線暫時(shí)硬編碼
Long companyId = 1L;
return courseBaseInfoService.createCourseBase(companyId,addCourseDto);
}如果校驗(yàn)出錯(cuò)Spring會(huì)拋出MethodArgumentNotValidException異常,我們需要在統(tǒng)一異常處理器中捕獲異常,解析出異常信息。
代碼 如下:
@ResponseBody
@ExceptionHandler(value = MethodArgumentNotValidException.class)
@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
public RestErrorResponse doValidException(MethodArgumentNotValidException argumentNotValidException) {
BindingResult bindingResult = argumentNotValidException.getBindingResult();
StringBuffer errMsg = new StringBuffer();
List<FieldError> fieldErrors = bindingResult.getFieldErrors();
fieldErrors.forEach(error -> {
errMsg.append(error.getDefaultMessage()).append(",");
});
log.error(errMsg.toString());
return new RestErrorResponse(errMsg.toString());
}重啟內(nèi)容管理服務(wù)。
使用httpclient進(jìn)行測(cè)試,將必填項(xiàng)設(shè)置為空,“適用人群” 屬性的內(nèi)容設(shè)置1個(gè)字。
執(zhí)行測(cè)試,接口響應(yīng)結(jié)果如下:
{
"errMessage": "課程名稱不能為空 課程分類不能為空 課程分類不能為空 適用人群內(nèi)容過(guò)少 "
}
可以看到校驗(yàn)器生效。
4.7.3 分組校驗(yàn)
有時(shí)候在同一個(gè)屬性上設(shè)置一個(gè)校驗(yàn)規(guī)則不能滿足要求,比如:訂單編號(hào)由系統(tǒng)生成,在添加訂單時(shí)要求訂單編號(hào)為空,在更新 訂單時(shí)要求訂單編寫不能為空。此時(shí)就用到了分組校驗(yàn),同一個(gè)屬性定義多個(gè)校驗(yàn)規(guī)則屬于不同的分組,比如:添加訂單定義@NULL規(guī)則屬于insert分組,更新訂單定義@NotEmpty規(guī)則屬于update分組,insert和update是分組的名稱,是可以修改的。
下邊舉例說(shuō)明
我們用class類型來(lái)表示不同的分組,所以我們定義不同的接口類型(空接口)表示不同的分組,由于校驗(yàn)分組是公用的,所以定義在 base工程中。如下:
package com.xuecheng.base.execption;
/**
* @description 校驗(yàn)分組
* @author Mr.M
* @date 2022/9/8 15:05
* @version 1.0
*/
public class ValidationGroups {
public interface Inster{};
public interface Update{};
public interface Delete{};
}下邊在定義校驗(yàn)規(guī)則時(shí)指定分組:
@NotEmpty(groups = {ValidationGroups.Inster.class},message = "添加課程名稱不能為空")
@NotEmpty(groups = {ValidationGroups.Update.class},message = "修改課程名稱不能為空")
// @NotEmpty(message = "課程名稱不能為空")
@ApiModelProperty(value = "課程名稱", required = true)
private String name;在Controller方法中啟動(dòng)校驗(yàn)規(guī)則指定要使用的分組名:
@ApiOperation("新增課程基礎(chǔ)信息")
@PostMapping("/course")
public CourseBaseInfoDto createCourseBase(@RequestBody @Validated({ValidationGroups.Inster.class}) AddCourseDto addCourseDto){
//機(jī)構(gòu)id,由于認(rèn)證系統(tǒng)沒(méi)有上線暫時(shí)硬編碼
Long companyId = 1L;
return courseBaseInfoService.createCourseBase(companyId,addCourseDto);
}再次測(cè)試,由于這里指定了Insert分組,所以拋出 異常信息:添加課程名稱不能為空。
如果修改分組為ValidationGroups.Update.class,異常信息為:修改課程名稱不能為空。
4.7.4 校驗(yàn)規(guī)則不滿足?
如果javax.validation.constraints包下的校驗(yàn)規(guī)則滿足不了需求怎么辦?
1、手寫校驗(yàn)代碼 。
2、自定義校驗(yàn)規(guī)則注解。
如何自定義校驗(yàn)規(guī)則注解,請(qǐng)自行查閱資料實(shí)現(xiàn)。
到此這篇關(guān)于springboot整合JSR303校驗(yàn)的文章就介紹到這了,更多相關(guān)springboot整合JSR303校驗(yàn)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Springboot集成JSR303參數(shù)校驗(yàn)的方法實(shí)現(xiàn)
- Spring Boot利用JSR303實(shí)現(xiàn)參數(shù)驗(yàn)證的方法實(shí)例
- SpringBoot結(jié)合JSR303對(duì)前端數(shù)據(jù)進(jìn)行校驗(yàn)的示例代碼
- SpringBoot使用jsr303校驗(yàn)的實(shí)現(xiàn)
- Spring中使用JSR303請(qǐng)求約束判空的實(shí)現(xiàn)
- SpringBoot后端進(jìn)行數(shù)據(jù)校驗(yàn)JSR303的使用詳解
- springboot接口參數(shù)校驗(yàn)JSR303的實(shí)現(xiàn)
- springboot整合JSR303參數(shù)校驗(yàn)與全局異常處理的方法
- SpringMVC中的JSR303與攔截器的使用方法
相關(guān)文章
Java實(shí)現(xiàn)簡(jiǎn)單的迷宮游戲詳解
迷宮游戲作為經(jīng)典的小游戲,一直深受大家的喜愛(ài)。本文小編將為大家詳細(xì)介紹一下如何用Java實(shí)現(xiàn)一個(gè)簡(jiǎn)單的迷宮小游戲,感興趣的可以動(dòng)手試一試2022-02-02
Java語(yǔ)言的Comparable和Comparator區(qū)別
這篇文章主要介紹了Java語(yǔ)言的Comparable和Comparator區(qū)別,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-06-06
spring-cloud入門之eureka-client(服務(wù)注冊(cè))
本篇文章主要介紹了spring-cloud入門之eureka-client(服務(wù)注冊(cè)),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-01-01
SpringBoot2底層注解@ConfigurationProperties配置綁定
這篇文章主要介紹了SpringBoot2底層注解@ConfigurationProperties配置綁定,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05
實(shí)例詳解java Struts2的配置與簡(jiǎn)單案例
這篇文章主要介紹了java Struts2的配置與簡(jiǎn)單案例,需要的朋友可以參考下2017-04-04

