91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

SpringBoot參數如何校驗

發布時間:2022-02-24 09:51:24 來源:億速云 閱讀:184 作者:iii 欄目:開發技術

本文小編為大家詳細介紹“SpringBoot參數如何校驗”,內容詳細,步驟清晰,細節處理妥當,希望這篇“SpringBoot參數如何校驗”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學習新知識吧。

一、前言

在 Web 開發中經常需要對前端傳過來的參數進行校驗,例如格式校驗、非空校驗等,基本上每個接口都需要進行校驗。如果使用常規的 IF ELSE 進行校驗,隨著參數越來越多,校驗邏輯的冗余度也越來越高,導致維護性變差。在 Java 中定義了一套基于注解的數據校驗規范 Bean Validation ,通過一些簡單的注解就能完成必要的邏輯校驗,相對來說就方便了很多。而 Bean Validation 只是規范,并沒有具體的實現,Hibernate 提供了具體的實現,也即 Hibernate Validator ,這個也是目前使用得比較多的驗證器了。

二、注解介紹

validator 內置注解

  • @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 被注釋的字符串的大小必須在指定的范圍內

  • @NotEmpty 被注釋的字符串的必須非空

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

  • @NotBlank 驗證字符串非 null ,且長度必須大于0

注意

  • @NotNull 用于驗證對象是否不為 null ,無法檢測長度為0的字符串;

  • @NotEmpty 用于 String、Map 或者數組等集合類型不能為 null 且長度必須大于0;

  • @NotBlank 只能用于String,不能為 null ,且調用 trim() 后,長度必須大于0;

校驗字符串是否為空,使用 @NotNull ,只有參數不傳的時候才會檢測到,傳了空值(例如空字符串)仍然可以通過校驗,因此應該使用 @NotBlank

三、添加依賴

SpringBoot 中 Bean Validation 已經集成在 starter-web 中,因此無需再添加依賴。但是本人實際測試發現,直接使用好像不行,因此添加了如下依賴:

<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-validation</artifactId>
</dependency>

四、創建用于校驗的實體類

創建一個 validator 目錄,在里面創建一個 UserDTO

本來想直接復用之前創建的 entity 類,但是后來想了下,entity 用于建立數據庫的映射關系,字段跟數據表是一一對應的,而這里的 validator 是用于校驗前端傳過來的參數,字段跟前端傳的參數是對應的,因此不能復用,需要單獨寫一個 validator 類。

順便提一下,在 RestController 中使用自己定義的對象,需要有 setter、getter 之類的方法,或者使用 lombok 的 @Data 注解。如果不加的話會報錯:

No converter found for return value of type: class validator.UserDTO

使用 getter、setter 方法如下:

public class UserDTO {
	private String username;
	private Integer age;

	public String getUsername() {
		return username;
	}
	public void setUsername(String username) {
		this.username = username;
	}

	public Integer getAge() {
		return age;
	}
	public void setAge(Integer age) {
		this.age = age;
	}
}

使用 lombok 的 @Data 注解如下,代碼與上面等效:

@Data
public class UserDTO {
	private String username;
	private Integer age;
}

五、寫一個測試用的接口

添加一個 POST 接口,從請求體中獲取參數,然后原封不動返回過去(主要是用來測試參數校驗的,這里接口邏輯并不重要)

@PostMapping("validateUser")
public UserDTO userValidate(@RequestBody UserDTO userDTO) {
    return userDTO;
}

六、在實體類中添加注解

給需要校驗的參數添加注解:

@Data
public class UserDTO {
    @NotBlank(message = "用戶名不能為空")
    private String username;

    @NotBlank(message = "手機號不能為空")
    private String mobile;

    @NotNull(message = "性別不能為空")
    private Integer sex;

    @NotNull(message = "年齡不能為空")
    private Integer age;

    @NotBlank(message = "郵箱不能為空")
    @Email(message = "郵箱格式錯誤")
    private String email;
}

七、在 controller 方法中添加 Validated 注解

然后需要在 controller 方法體添加 @Validated ,不加 @Validated 校驗會不起作用。

用下面的數據測試一下:

{
    "username": "",
    "mobile": "2333",
    "sex": 0,
    "age": 0,
    "email": "233@dby.com"
}

可以看到校驗是成功了,但是后臺拋了一個異常:

Validation failed for argument [0] in public validator.UserDTO com.hhlnyfz.hhlnyfz.HelloController.userValidate(validator.UserDTO): [Field error in object ‘userDTO' on field ‘username': rejected value []; codes [NotBlank.userDTO.username,NotBlank.username,NotBlank.java.lang.String,NotBlank]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [userDTO.username,username]; arguments []; default message [username]]; default message [用戶名不能為空]] ]

然后返回參數并不理想,前端也并不容易處理返回參數

八、添加全局異常處理

上面這種情況需要添加一下全局異常處理,這樣比較規范。創建一個 GlobalExceptionHandler 類,在類的上面添加 @RestControllerAdvice 注解,然后添加如下代碼:

/**
 * 全局異常處理類
 */
@RestControllerAdvice
public class GlobalExceptionHandler {
    // 捕獲 MethodArgumentNotValidException 異常
    @ExceptionHandler(value = MethodArgumentNotValidException.class)
    public HashMap<String, Object> handleMethodArgumentNotValidException(MethodArgumentNotValidException e, HttpServletRequest request) {
        HashMap<String, Object> map = new HashMap<>();
        map.put("code", 400);
        map.put("msg", e.getMessage());
        map.put("url", request.getRequestURL());
        return map;
    }

    // 其他異常處理方法
}

這邊 @ExceptionHandler 注解中的 MethodArgumentNotValidException.class 用于捕獲請求參數異常。如果是 Exception.class 表示捕獲全部異常。不要用一個方法處理所有的異常,而是一個方法處理一種異常。如果需要處理其他異常,可以在下面添加方法。

還是用剛才的測試用例,這次異常被捕獲到了

可以看到 e.getMessage() 把整個錯誤堆棧信息全部打印出來了,但我們只需要把最后的 default message 返回給前端就行,因此改用 e.getBindingResult().getFieldError().getDefaultMessage() ,然后 IDE 給了提示:

Method invocation ‘getDefaultMessage' may produce ‘NullPointerException'

也就是說 e.getBindingResult().getFieldError() 可能會是一個空指針 null ,于是按照 IDE 的提示用 Objects.requireNonNull 包裹一下,最終代碼如下:

/**
 * 全局異常處理類
 */
@RestControllerAdvice
public class GlobalExceptionHandler {
    // 捕獲 MethodArgumentNotValidException 異常
    @ExceptionHandler(value = MethodArgumentNotValidException.class)
    public HashMap<String, Object> handleMethodArgumentNotValidException(MethodArgumentNotValidException e, HttpServletRequest request) {
        HashMap<String, Object> map = new HashMap<>();
        map.put("code", 400);
        map.put("msg", Objects.requireNonNull(e.getBindingResult().getFieldError()).getDefaultMessage());
        map.put("url", request.getRequestURL());
        return map;
    }

    // 其他異常處理方法
}

當然這邊還是有不規范的地方,沒有用統一響應體進行返回,后面會介紹如何封裝統一響應體。

九、分組校驗

@Valid@Validated 兩個注解都可以實現校驗,前面的功能用 @Valid 也是可以的,但是 @Validated 功能更強大,可以實現分組校驗。什么是分組校驗,分組校驗實際上實現了實體類的復用,有時候并不希望對所有的參數都進行校驗,例如下面這個情況:

@Data
public class Route {
	@NotNull(message = "始發地省id不能為空")
	private Integer startProvinceId;
	
	@NotNull(message = "目的地省id不能為空")
	private Integer endProvinceId;
	
	@NotBlank(message = "詳細地址不能為空")
	private String address;
}

假如在一個接口中只希望校驗 startProvinceIdaddress ,而在另一個接口中只希望校驗 endProvinceIdaddress ,這個時候就可以用分組校驗。可以定義一個接口:

public interface ValidateGroup {
	interface RouteValidStart {}
	interface RouteValidEnd {}
}

然后在實體類中添加分組:

@Data
public class Route {
	@NotNull(groups = {RouteValidStart.class}, message = "始發地省id不能為空")
	private Integer startProvinceId;
	
	@NotNull(groups = {RouteValidEnd.class}, message = "目的地省id不能為空")
	private Integer endProvinceId;
	
	@NotBlank(groups = {RouteValidStart.class, RouteValidEnd.class}, message = "詳細地址不能為空")
	private String address;
}

然后在校驗的時候只需要把分組傳入 @Validate 就可以實現指定參數的校驗:

@RequestMapping("addRoute")
public ServerResponse addRoute(@RequestBody @Validated({RouteValidStart.class}) Route route) {
	// ...
	return ServerResponse.success();
}

十、單個參數校驗

在參數前面加上注解即可:

@PostMapping("/get")
public ReturnVO getUserInfo(@RequestParam("userId") @NotNull(message = "用戶ID不能為空") String userId){
    return new ReturnVO().success();
}

然后在 Controller 類上面增加 @Validated 注解,注意不是增加在參數前面。

讀到這里,這篇“SpringBoot參數如何校驗”文章已經介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領會,如果想了解更多相關內容的文章,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

平安县| 中阳县| 桐柏县| 北辰区| 江西省| 沿河| 自治县| 兰州市| 仙桃市| 城步| 滨海县| 余庆县| 陵川县| 深圳市| 临泽县| 太谷县| 青田县| 湖南省| 桂东县| 胶南市| 万安县| 台前县| 宝丰县| 龙川县| 大连市| 广灵县| 皋兰县| 云南省| 三亚市| 济宁市| 利辛县| 崇明县| 泾川县| 靖西县| 甘孜| 郯城县| 民勤县| 五大连池市| 察隅县| 辉县市| 沐川县|