国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

優雅的springboot參數校驗,你學會了嗎?

開發 前端
分享的這篇文章里的校驗參數注解使用方法,我是在一個springboot項目里親自重新測試驗證過的,springboot的版本是2.3.9.RELEASE,另外也引入了關于參數校驗的starter包,這樣就不用額外去引關于參數校驗的其他包了。

前言

在后端的接口開發過程,實際上每一個接口都或多或少有不同規則的參數校驗,有一些是基礎校驗,如非空校驗、長度校驗、大小校驗、格式校驗;也有一些校驗是業務校驗,如學號不能重重復、手機號不能重復注冊等;對于業務校驗,是需要和數據庫交互才能知道校驗結果;對于參數的基礎校驗,是有一些共有特征可以抽象出來,可以做成一個通用模板(java就是一種面向對象的編程語言,還記得天天快要說爛問爛的面向對象的三大特性嗎?)。基于實際場景的需要,java API中定義了一些Bean校驗的規范標準(JSR303:validation-api),但是沒有具體實現,不過hibernate validation和spring validation都提供了一些比較優秀的實現。如果在項目里,你還是像類似這樣的方式來進行參數校驗就太low了,活該加班到天亮(當然如果你所在公司目前仍然用統計代碼量來考核你的工作,就算我沒說,你可以繼續使用這種方式)。

@PostMapping("/add")
public String add(Student student) {
    if (null == student) {
        throw new RuntimeException("學生不為空");
    }
    if ("".equals(student.getStuCode())) {
        throw new RuntimeException("學號不能為空");
    }
    if ("".equals(student.getStuName())) {
        throw new RuntimeException("學生姓名不能為空");
    }
    if (null == student.getTeacher()) {
        throw new RuntimeException("學生的老師的不能為空");
    }
    if ("".equals(student.getTeacher().getTecName())) {
        throw new RuntimeException("學生的老師的姓名不能為空");
    }
    if ("".equals(student.getTeacher().getSubject())) {
        throw new RuntimeException("學生的老師的所授科目不為能空");
    }
    return "success";
}

依賴引入

分享的這篇文章里的校驗參數注解使用方法,我是在一個springboot項目里親自重新測試驗證過的,springboot的版本是2.3.9.RELEASE,另外也引入了關于參數校驗的starter包,這樣就不用額外去引關于參數校驗的其他包了;

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-validation</artifactId>
  <version>2.3.9.RELEASE</version>
</dependency>

參數形式

在java項目中,前端請求后端的接口中,常用的請求類型主要是post和get。

  • 在POST請求中,通常使用requestBody傳遞參數,即前端以json報文的格式傳遞到后端controller層,spring會把json報文自動映射到@RequestBody修飾的形參實例;
  • 在GET請求中,通常使用requestParam/PathVariable傳遞參數,其中requestParam是指前端以key-value的形式把參數傳遞到后端,spring會把參數自動映射到@RequestParam修飾的形參數實例對象(@RequestParam可以,也可以沒有,只要參數key與controller層方法內形參類型的屬性名稱可以對應的上);@PathVariable是指spring可以將請求URL中占位符參數綁定到controller層方法的形參上;

常用到的約束注解

@Valid

被注釋的元素是一個對象,需要檢查此對象的所有字段值

@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(min=, max=)

被注釋的字符串的大小必須在指定的范圍內

@NotEmpty

被注釋的字符串的必須非空

@Range(min=, max=)

被注釋的元素必須在合適的范圍內

@NotBlank

被注釋的字符串的必須非空

@URL(protocol=,

host=,    port=, 

regexp=, flags=)

被注釋的字符串必須是一個有效的url

@CreditCardNumber

被注釋的字符串必須通過Luhn校驗算法,

銀行卡,信用卡等號碼一般都用Luhn

計算合法性

@ScriptAssert

(lang=, script=, alias=)

要有Java Scripting API 即JSR 223 

("Scripting for the JavaTM Platform")的實現

@SafeHtml

(whitelistType=, 

additionalTags=)

classpath中要有jsoup包

參數基礎校驗

參數的基礎校驗,通常是指的非空、長度、最大值、最小值、格式(數字、郵箱、正則)等這些場景的校驗。

@RequestBody參數

1.在controller層的方法的形參數前面加一個@Valid或@Validated的注解;

2.在用@RequestBody修飾的類的屬性上加上約束注解,如@NotNull、@Length、@NotBlank;

3.@RequestBody參數在觸發校驗規則時,會拋出MethodArgumentNotValidException,這里使用統一的異常處理機制來處理異常;

總結:第1步的valid的作用就是一個標記,標明這個參數需要進行校驗;第2步的約束注解的上注明校驗的規則;第3步的統一校驗機制是前后臺請求后臺接口時,如果校驗參數的校驗規則后會拋出異常,異常附帶有約束注解上的提示信息,那么通過異常統一處理機制就可以統一處理異常信息,并以合適的方式返回給前臺(所謂合適的方式是指異常信息的格式可以自行制定)。

@PostMapping("/add")
public Student add( @Valid@RequestBody Student student){
    System.out.println(student.getStuName());
    return student;
}
@Data
public class Student  {
    @NotNull(message = "學號不能為空")
    @Length(min = 2, max = 4, message = "學號的長度范圍是(2,4)")
    private String stuCode;
    @NotNull(message = "姓名不能為空")
    @Length(min = 2, max = 3, message = "姓名的長度范圍是(2,3)")
    private String stuName;
}

@RequestParam參數/@PathVariable參數

1.在controller層的控制類上添加@Validated注解;

2.在controller層方法的校驗參數上添加約束注解,如@NotNull、@Pattern;

3.@RequestParam參數/@PathVariable參數在觸發校驗規則時,會拋出ConstraintViolationException類型的異常,所以在統一異常處理機制中添加對這種類型異常的處理機制;

@RestController
@RequestMapping("/student")
@Validated
public class StudentController {
    @GetMapping("/{sex}/info")
    public String getBySex(@PathVariable("sex") @Pattern(regexp = "boy||girl",message = "學生性別只能是boy或girl") String sex) {
        System.out.println("學生性別:" + sex);
    return "success";
    }
    @GetMapping("/getOne")
    public String getOne(@NotNull(message = "學生姓名不能為空") String stuName, @NotNull(message = "學生學號不能為空") String stuCode) {
        System.out.println("stuName:" + stuName + ",stuCode:" + stuCode);
    return "success";
    }
}

異常統一處理

@RestControllerAdvice
public class CommonExceptionHandler {


    /**
     * 用于捕獲@RequestBody類型參數觸發校驗規則拋出的異常
     *
     * @param e
     * @return
     */
    @ExceptionHandler(value = MethodArgumentNotValidException.class)
    public String handleValidException(MethodArgumentNotValidException e) {
        StringBuilder sb = new StringBuilder();
        List<ObjectError> allErrors = e.getBindingResult().getAllErrors();
        if (!CollectionUtils.isEmpty(allErrors)) {
            for (ObjectError error : allErrors) {
                sb.append(error.getDefaultMessage()).append(";");
            }
        }
        return sb.toString();
    }


    /**
     * 用于捕獲@RequestParam/@PathVariable參數觸發校驗規則拋出的異常
     *
     * @param e
     * @return
     */
    @ExceptionHandler(value = ConstraintViolationException.class)
    public String handleConstraintViolationException(ConstraintViolationException e) {
        StringBuilder sb = new StringBuilder();
        Set<ConstraintViolation<?>> conSet = e.getConstraintViolations();
        for (ConstraintViolation<?> con : conSet) {
            String message = con.getMessage();
            sb.append(message).append(";");
        }
        return sb.toString();
    }
}

嵌套校驗

在實際項目中有這樣一種場景,用來接收參數的類的屬性字段也是一個對象,屬性對象的字段也需要進行必要的參數校驗,這個時候可以使用嵌套校驗來解決這個問題,hibernate-validator提供了具體的解決方式。

1.在controller層方法的形參數前添加@Validated注解,如果有分組校驗的場景,則注明分組信息;如果校驗不需要分組,可以不注明分組信息;

2.在接收參數的類的屬性是對象的字段上添加@Valide注解,這里需要注意的是一定是@Valid,不是@Validated,因為@Valid的實現是由hibernate-validator提供,有嵌套校驗的能力,而@Validated是由spring-validation提供的具體實現方式,@Validated有分組校驗的能力,但是沒有嵌套校驗的能力;(java API規范(JSR303)定義了Bean的校驗標準validation-api,但是沒有具體的實現,所以各有各的實現,在功能上也是有區別的)

3.嵌套屬性類上的約束注解的用法,與用來接收參數的對象屬性上的約束注解的用法是一樣的;

總結:@Valid的實現是由hibernate-validator提供,有嵌套校驗的能力,但是沒有分組校驗的能力,@Validated是由spring-validation提供的具體實現方式,@Validated有分組校驗的能力,但是沒有嵌套校驗的能力,在使用的過程須特別注意,要根據實際需要進行剪裁。

@PostMapping("/addStuaAndTeach")
public String addStuaAndTeach(@Validated(AddStuAndTeach.class) @RequestBody Student student){
    System.out.println("學生的工號:"+student.getStuCode()+",學生的老師的姓名:"+student.getTeacher().getTecName());
    return "success";
}
@Data
public class Teacher {
    @NotNull(message = "學生的老師姓名不能為空",groups = AddStuAndTeach.class)
    private String tecName;
    @NotNull(message = "學生的老師教授科目不能為空",groups = AddStuAndTeach.class)
    private String subject;
}
public interface AddStuAndTeach {
}
@Data
public class Student {
   
    @NotNull(message = "學生id不能為空",groups = QueryDetail.class)
    private Integer id;
    @NotNull(message = "學號不能為空",groups = AddStudent.class)
    @Length(min = 2, max = 4, message = "學號的長度范圍是(2,4)")
    private String stuCode;
    @NotNull(message = "姓名不能為空",groups = AddStudent.class)
    @Length(min = 2, max = 3, message = "姓名的長度范圍是(2,3)",groups = AddStudent.class)
    private String stuName;
    @Valid
    @NotNull(message = "學生的老師不能為空",groups = AddStuAndTeach.class)
    private Teacher teacher;
}

分組校驗

在實際的項目中,可能多個方法使用同一個類來接收參數,但是不同的方法的校驗規則又是不同的,這個時候就可以使用分組校驗的方式來解決這個問題了,spring-validation提供了具體的實現方式。

1.聲明分組用的接口,比如添加和查詢詳情的時候,校驗的規則肯定是不一樣的,添加的時候一般不用傳id,由后臺自增長生成,查詢詳情的時候id是必須傳的;

2.在controller層方法的校驗參數上添加@Validated參數,同時注解里要注明校驗參數的分組信息;

3.在校驗參數的類上的線束注解上,也要注明校驗參數的分組信息;

總結:在接口的入口方法參數上、校驗參數上都注明了分組的信息,那么接口被用的時候,就可以根據不同的分組信息執行不同約束注解的校驗邏輯了,這個能力是spring-validation提供的,所以這種場景下,controller層方法的上注解要用@Validated,@Valid注解沒有這種能力。

//用于添加場景參數校驗分組
public interface AddStudent {
}
//用于查詢詳情場景參數校驗分組
public interface QueryDetail {
}
@PostMapping("/add")
public Student add(@Validated(AddStudent.class) @RequestBody Student student) {
    System.out.println(student.getStuName());
    return student;
}


@PostMapping("/detail")
public String detail(@Validated(QueryDetail.class)@RequestBody Student student){
    System.out.println("學生id:"+student.getId());
    return "success";
}
@Data
public class Student {
 
    @NotNull(message = "學生id不能為空",groups = QueryDetail.class)
    private Integer id;
    @NotNull(message = "學號不能為空",groups = AddStudent.class)
    @Length(min = 2, max = 4, message = "學號的長度范圍是(2,4)")
    private String stuCode;
    @NotNull(message = "姓名不能為空",groups = AddStudent.class)
    @Length(min = 2, max = 3, message = "姓名的長度范圍是(2,3)",groups = AddStudent.class)
    private String stuName;
    }


責任編輯:武曉燕 來源: 凡夫編程
相關推薦

2024-05-29 07:47:30

SpringJava@Resource

2023-08-08 08:23:08

Spring日志?線程池

2023-11-27 07:26:42

Springboot容器

2022-07-08 09:27:48

CSSIFC模型

2024-01-19 08:25:38

死鎖Java通信

2023-01-10 08:43:15

定義DDD架構

2024-02-04 00:00:00

Effect數據組件

2023-07-26 13:11:21

ChatGPT平臺工具

2024-02-02 11:03:11

React數據Ref

2024-01-02 12:05:26

Java并發編程

2023-08-01 12:51:18

WebGPT機器學習模型

2023-12-05 17:57:13

nginx參數

2024-10-17 10:00:59

2024-05-07 07:58:47

C#程序類型

2025-01-07 08:37:35

2023-01-30 09:01:54

圖表指南圖形化

2024-05-06 00:00:00

InnoDBView隔離

2023-12-12 08:02:10

2024-07-31 08:39:45

Git命令暫存區

2023-10-10 11:04:11

Rust難點內存
點贊
收藏

51CTO技術棧公眾號

**国产精品| 欧美这里只有精品| av有码在线观看| 欧美性20hd另类| 天堂色在线视频| 国产亚洲精久久久久久| 欧美激情亚洲天堂| 精品一区精品二区高清| 99在线影院| 欧美gayvideo| 久久69精品久久久久久国产越南| 欧美激情网站| 91精品视频网| 香蕉av一区| 99九九热只有国产精品| 99久久自偷自偷国产精品不卡| 亚洲日韩第一页| 毛片视频免费观看| 欧美激情一区二区三区不卡| 国产自产在线视频| 99免费精品在线| www.黄色网址.com| 成人av手机在线观看| 97在线国产视频| 91亚洲大成网污www| 国模杨依粉嫩蝴蝶150p| 久久美女高清视频| 一区二区三区网址| 亚洲精选视频免费看| 国精产品999国精产品官网| 亚洲精品乱码久久久久久久久| 狠狠操第一页| 亚洲成人激情综合网| 青青青草原在线| 日韩视频一区二区| 午夜欧美巨大性欧美巨大| 色阁综合伊人av| 色噜噜成人av在线| 久久视频国产精品免费视频在线| 日本精品在线播放| 国产成人精品在线视频| 欧美色女视频| 国产精品区一区二区三含羞草| 一本色道久久综合亚洲精品不卡| 亚洲精品一区二区三区蜜桃久 | 青青久久精品| 亚洲大奶少妇| 久久亚洲欧美日韩精品专区| 一级毛片aaaaaa免费看| 国产高清在线看| 欧美在线一区二区| 1区2区3区在线| 色综久久综合桃花网| 精品国产导航| av免费精品一区二区三区| 精品一区二区三区欧美| 在线黄色免费观看| 欧美丝袜丝交足nylons图片| 69久成人做爰电影| 91国产美女视频| 六月天综合网| 免费男女羞羞的视频网站中文版 | 欧美系列一区二区| 中文字幕在线免费观看视频| 精品少妇v888av| 欧美精品黄色| 美女日批免费视频| 一道本成人在线| 亚洲成人va| 亚洲aa中文字幕| 成人精品鲁一区一区二区| 天堂电影在线| 国产亚洲欧洲在线| 不卡一区2区| 男人添女人下部视频免费| 亚洲一卡二卡三卡四卡无卡久久| 国产精品沙发午睡系列| 天天影视涩香欲综合网| av剧情在线观看| 国产成人精品在线视频| 欧美国产综合色视频| 99精品视频网| 成人另类视频| 999精品网| 免费在线看黄色| 欧美男人的天堂一二区| 国产自产v一区二区三区c| 中文亚洲字幕| 99国产精品久久久久久久成人热| 成人搞黄视频| 日本片在线观看| 国产黑人绿帽在线第一区| 久久久蜜桃一区二区人| 成人综合视频在线| 亚洲成色777777女色窝| 日本在线免费播放| 成人综合av网| 97国产一区二区| 中文av在线播放| 一本一本久久a久久精品综合小说 一本一本久久a久久精品牛牛影视 | 亚洲一区三区视频在线观看| 人人草在线视频| 97久久精品人人澡人人爽| 一级片免费在线观看| 日韩在线播放视频| 免费看黄色91| 日本在线免费网| 清纯唯美亚洲综合| 亚洲精品亚洲人成人网| 色综合久久中文| 一本色道久久综合亚洲精品按摩| 高清不卡一区| 国产精品12p| 日韩一区二区电影在线| 欧美激情第8页| 大地资源高清播放在线观看| 欧美精品久久一区二区 | 国产精品久久不能| 国产午夜亚洲精品不卡| 欧美男体视频| 四虎免费在线观看视频| 欧美大片国产精品| 久久精品91| www视频在线看| 麻豆精品传媒视频| 欧美日本不卡视频| 亚洲影院一区| 一区二区三区伦理| 欧美一级二级三级| 精品久久久久久久久久久院品网| 久久字幕精品一区| 国产第一页在线视频| 亚洲欧美日韩国产yyy| 亚洲精品av在线| 国产精品18久久久| 日韩欧美一区二区三区免费观看 | 亚洲精品乱码| 麻豆视频在线观看免费网站| 国产在线欧美日韩| 欧美一级日韩不卡播放免费| 影音先锋久久| 丰满诱人av在线播放| 中文字幕乱码一区二区三区 | 国产麻豆精品在线| 欧美电影免费观看| 丰满少妇大力进入| 九九热精品视频国产| 国产精品乱人伦中文| 日韩一级电影| 免费黄网站在线观看| 欧美成人综合一区| 亚洲品质视频自拍网| 96av麻豆蜜桃一区二区| 久久久久97| 你懂的在线看| 亚洲精品中文字幕乱码三区不卡| 亚洲老头同性xxxxx| www.亚洲精品| 久久亚洲黄色| 九色视频网站在线观看| 日韩欧美视频一区二区| 精品国产拍在线观看| 亚洲精品视频在线看| 999在线观看精品免费不卡网站| 阿v视频在线观看| 免费涩涩18网站入口| 91原创国产| 伊人久久综合97精品| 亚洲一区视频在线| 日本不卡视频一二三区| 久久久久毛片免费观看| 亚洲精华国产精华| 日韩精品久久久免费观看| 在线视频一区二区| 综合自拍亚洲综合图不卡区| 欧美日韩影院| 大胆国模一区二区三区| 天堂中文在线资| 国产免费xxx| 国产成人精品电影| 精品国内片67194| 国产精品久久久久影视| 免费在线亚洲欧美| 色吊丝一区二区| 色yeye免费人成网站在线观看| 在线免费av播放| 欧美成人dvd在线视频| 久久久视频在线| 日韩一区二区三区在线| 国产精品每日更新| 蜜桃精品在线观看| 人人狠狠综合久久亚洲婷| 黄色18在线观看| 青檬在线电视剧在线观看| 久久久久久久久久久久久国产| 国产精品久久中文| 国产一区二区三区在线观看视频 | 国产字幕视频一区二区| 免费一区二区三区四区| 亚洲成人三级|