欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SpringBoot如何優(yōu)雅實現(xiàn)接口參數(shù)驗證

 更新時間:2023年08月25日 08:48:22   作者:小虎哥的技術(shù)博客  
為了保證參數(shù)的正確性,我們需要使用參數(shù)驗證機制,來檢測并處理傳入的參數(shù)格式是否符合規(guī)范,所以本文就來和大家聊聊如何優(yōu)雅實現(xiàn)接口參數(shù)驗證吧

1. 為什么需要參數(shù)驗證

在應(yīng)用程序的開發(fā)中,我們經(jīng)常會遇到需要保證傳入?yún)?shù)的正確性的情況。例如,當我們在注冊用戶時,需要驗證用戶填寫的表單數(shù)據(jù)是否符合規(guī)范,是否缺少必填字段,或者格式是否正確,等等。如果不對參數(shù)進行驗證,我們的應(yīng)用程序可能會因此受到攻擊或者運行出錯。

為了保證參數(shù)的正確性,我們需要使用參數(shù)驗證機制,來檢測并處理傳入的參數(shù)格式是否符合規(guī)范。

2. 如何進行參數(shù)驗證

Spring Boot內(nèi)置了一個很強大的參數(shù)驗證框架——JSR 303 Bean Validation 標準,它可以對我們的實體類參數(shù)進行校驗,并且可以給我們提供詳細的錯誤提示信息。

具體步驟如下:

1. 添加依賴

pom.xml 文件中添加以下依賴:

<!-- 添加 JSR-303 Bean Validation 依賴 -->
<dependency>
    <groupId>javax.validation</groupId>
    <artifactId>validation-api</artifactId>
    <version>2.0.1.Final</version>
</dependency>
<!-- Hibernate Validator 的實現(xiàn) --> 
<dependency>
    <groupId>org.hibernate.validator</groupId>
    <artifactId>hibernate-validator</artifactId>
    <version>6.1.5.Final</version>
</dependency>

2. 在實體類中添加參數(shù)驗證注解

public class User {
    @NotNull(message="用戶名不能為空")
    private String username;
    @NotNull(message="密碼不能為空")
    @Size(min=6, max=20, message="密碼長度應(yīng)該在6到20位")   
    private String password;
    // getter 和 setter 方法
}

在實體類中使用注解,如上代碼所示,可以指定當前字段是否可以為 null,以及字符串的長度和格式等信息,如果傳入的參數(shù)不符合預(yù)定的規(guī)則,在校驗時就會產(chǎn)生相應(yīng)的提示。

常用的參數(shù)驗證注解:

  • @NotNull:驗證注解的元素值不為null;
  • @NotEmpty:驗證注解的元素值不為null,且String類型也不為 "";
  • @NotBlank:驗證注解的元素值不為null,且允許去除兩端空格后不為空;
  • @Size:驗證注解的元素值長度在min和max范圍內(nèi);
  • @Digits:驗證注解元素值的整數(shù)位數(shù)和小數(shù)位數(shù)是否符合預(yù)期;
  • @Range:驗證注解元素值的大小是否在指定范圍內(nèi);
  • @Email:驗證注解的元素值是否為Email格式。

3. 在Controller中加入對參數(shù)的驗證

@RestController
@RequestMapping("/user")
public class UserController {
    @PostMapping("/register")
    public CommonResult<User> register(@RequestBody @Validated User user) {
        return CommonResult.success(user);
    }
}

在Controller層進行參數(shù)校驗時,加上 @Validated 注解以告訴Spring進行參數(shù)校驗,@RequestBody 注解用于將請求體中的JSON數(shù)據(jù)綁定到實體類中。

4. 定義全局異常處理器,并定義返回結(jié)果CommonResult

@RestControllerAdvice
public class GlobalExceptionHandler {
    @ExceptionHandler(value = MethodArgumentNotValidException.class)
    public CommonResult<Object> handleMethodArgumentNotValidException(MethodArgumentNotValidException ex){
        List<ObjectError> errors = ex.getBindingResult().getAllErrors();
        StringBuilder errorMsg = new StringBuilder();
        errors.forEach(error -> errorMsg.append(error.getDefaultMessage()).append("; "));
        return CommonResult.fail(ResultCode.PARAM_VALID_ERROR.getCode(), errorMsg.toString(), null);
    }
    @ExceptionHandler(value = Exception.class)
    public CommonResult<Object> handleException(Exception ex){
        log.error("系統(tǒng)發(fā)生異常:", ex);
        return CommonResult.fail(ResultCode.SYSTEM_ERROR.getCode(), ResultCode.SYSTEM_ERROR.getMsg(), null);
    }
}

在上述代碼中,我們定義了兩個異常處理方法,一個處理 MethodArgumentNotValidException 異常,一個處理其他類型的異常,使用了 @ExceptionHandler 注解將不同類型的異常映射到不同的處理方法中。

public enum ResultCode {
    SUCCESS(200, "操作成功"),
    PARAM_VALID_ERROR(400, "參數(shù)校驗失敗"),
    UNAUTHORIZED(401, "未認證"),
    FORBIDDEN(403, "無權(quán)限"),
    SYSTEM_ERROR(500, "服務(wù)器內(nèi)部錯誤");
    private int code;
    private String msg;
    ResultCode(int code, String msg) {
        this.code = code;
        this.msg = msg;
    }
    public int getCode() {
        return code;
    }
    public String getMsg() {
        return msg;
    }
}
public class CommonResult<T> {
    private int code;
    private String msg;
    private T data;
    public CommonResult() {}
    public CommonResult(int code, String msg, T data) {
        this.code = code;
        this.msg = msg;
        this.data = data;
    }
    public static <T> CommonResult<T> success(T data) {
        return new CommonResult<T>(ResultCode.SUCCESS.getCode(), ResultCode.SUCCESS.getMsg(), data);
    }
    public static <T> CommonResult<T> fail(ResultCode resultCode) {
        return new CommonResult<T>(resultCode.getCode(), resultCode.getMsg(), null);
    }
    public static <T> CommonResult<T> fail(ResultCode resultCode, T data) {
        return new CommonResult<T>(resultCode.getCode(), resultCode.getMsg(), data);
    }
    // getter 和 setter 方法
}

在上述代碼中,我們使用泛型和枚舉類定義各種操作返回的枚舉值,通過封裝返回值,返回給前端統(tǒng)一的返回格式,減少重復(fù)代碼。

5. 通過Postman模擬測試參數(shù)驗證

我們將請求方式設(shè)置為 POST,并將請求的 URL 設(shè)置為 http://localhost:8080/user/register。另外,我們還需要設(shè)置請求頭的 Content-Type 為 application/json。

我們傳入了一個用戶名為 user ,密碼為 123 的用戶并進行請求,這個請求在參數(shù)校驗的過程中會出現(xiàn)異常。

最終的打印結(jié)果為:

{
    "code": 400,
    "msg": "密碼長度應(yīng)該在6到20位; ",
    "data": null
}

3.自定義注解和驗證器

在實體類中使用注解進行參數(shù)驗證,這種方式非常簡單快捷,但對于一些復(fù)雜的邏輯驗證可能不夠靈活。我們也可以采用編寫自定義注解和驗證器的方式進行參數(shù)驗證。

比如,我們可以編寫自定義注解 @Age 和驗證器 AgeValidator,使用這兩個自定義類對請求參數(shù)進行驗證,示例代碼如下:

@Target({ElementType.FIELD, ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Constraint(validatedBy = {AgeValidator.class})
public @interface Age {
    String message() default "年齡不在合法范圍內(nèi)";
    int minAge() default 0;
    int maxAge() default 150;
    Class<?>[] groups() default {};
    Class<? extends Payload>[] payload() default {};
}
public class AgeValidator implements ConstraintValidator<Age, Integer> {
    private int minAge;
    private int maxAge;
    @Override
    public void initialize(Age constraintAnnotation) {
        this.minAge = constraintAnnotation.minAge();
        this.maxAge = constraintAnnotation.maxAge();
    }
    @Override
    public boolean isValid(Integer value, ConstraintValidatorContext context) {
        if(value == null) {
            return true; // 允許為空值
        }
        if(value < minAge || value > maxAge) {
            return false; // 驗證不通過
        }
        return true; // 驗證通過
    }
}
public class User {
    @NotNull(message="用戶名不能為空")
    private String username;
    @Age(message="年齡不在合法范圍內(nèi)", minAge=0, maxAge=120)
    private Integer age;
    // getter 和 setter 方法
}

如上代碼展示了如何使用自定義注解和驗證器進行參數(shù)驗證,使用自定義注解時,我們可以指定年齡的最小值和最大值,并使用 AgeValidator 在實體類中對年齡進行驗證。

4.小結(jié)

使用參數(shù)驗證機制可以有效地提高我們應(yīng)用程序的健壯性和安全性,同時也可以節(jié)省我們進行參數(shù)校驗的時間和精力。Spring Boot 內(nèi)置的參數(shù)驗證框架 JSR 303 Bean Validation 標準可以幫助我們簡單快速地完成參數(shù)校驗,結(jié)合全局異常處理器和自定義的響應(yīng)結(jié)果類,我們可以快速反饋請求信息的正確性,并在出現(xiàn)錯誤時提供詳細的錯誤提示信息,非常方便實用。

以上就是SpringBoot如何優(yōu)雅實現(xiàn)接口參數(shù)驗證的詳細內(nèi)容,更多關(guān)于SpringBoot接口參數(shù)驗證的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • SpringBoot開發(fā)技巧之使用AOP記錄日志示例解析

    SpringBoot開發(fā)技巧之使用AOP記錄日志示例解析

    這篇文章主要為大家介紹了SpringBoot開發(fā)技巧之如何利用AOP記錄日志的示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步
    2021-10-10
  • 兩種實現(xiàn)Java類隔離加載的方法

    兩種實現(xiàn)Java類隔離加載的方法

    這篇文章主要介紹了兩種實現(xiàn)Java類隔離加載的方法,幫助大家更好的理解和學(xué)習(xí)使用Java,感興趣的朋友可以了解下
    2021-02-02
  • idea同時編輯多行問題-win&mac都支持

    idea同時編輯多行問題-win&mac都支持

    這篇文章主要介紹了idea同時編輯多行問題-win&mac都支持,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • 詳解Java實踐之抽象工廠模式

    詳解Java實踐之抽象工廠模式

    抽象工廠模式用于產(chǎn)品族的構(gòu)建。抽象工廠是所有形態(tài)的工廠模式中最為抽象和最具一般性的一種形態(tài)。抽象工廠是指當有多個抽象角色時使用的一種工廠模式。抽象工廠模式可以向客戶端提供一個接口,使客戶端在不必指定產(chǎn)品的具體情況下,創(chuàng)建多個產(chǎn)品族中的產(chǎn)品對象
    2021-06-06
  • Idea 搭建Spring源碼環(huán)境的超詳細教程

    Idea 搭建Spring源碼環(huán)境的超詳細教程

    這篇文章主要介紹了Idea 搭建Spring源碼環(huán)境,本文通過圖文并茂的形式給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-10-10
  • 通過實例解析java String不可變性

    通過實例解析java String不可變性

    這篇文章主要介紹了通過實例解析java String不可變性,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-03-03
  • 利用Java自寫一個生成ID的工具類

    利用Java自寫一個生成ID的工具類

    平時項目中只要涉及表,那么一定能接觸到眾多各式各樣的ID編號。本文將通過Java語言實現(xiàn)手寫一個ID生成工具類,需要的小伙伴可以參考一下
    2022-11-11
  • 如何重寫hashcode和equals方法

    如何重寫hashcode和equals方法

    這篇文章主要介紹了如何重寫hashcode和equals方法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-06-06
  • SpringBoot整合Spring Batch示例代碼

    SpringBoot整合Spring Batch示例代碼

    這篇文章主要來和大家一起探討一下SpringBoot如何整合Spring Batch,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-10-10
  • Java多線程之線程狀態(tài)詳解

    Java多線程之線程狀態(tài)詳解

    這篇文章主要介紹了Java多線程 線程狀態(tài)原理詳解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2021-09-09

最新評論