Spring 中@Validated 分組校驗的使用解析

Spring @Validated分組校驗的使用

通過本文你能學習到@Validated 的基本使用,以及如何再spring-boot 中進行數據異常的統一處理

Spring Validation驗證框架對參數的驗證機制提供瞭@Validated(Spring’s JSR-303規范,是標準JSR-303的一個變種),javax提供瞭@Valid(標準JSR-303規范),配合BindingResult可以直接提供參數驗證結果。

在檢驗入參是否符合規范時,使用@Validated或者@Valid在基本驗證功能上沒有太多區別。但是在分組、註解地方、嵌套驗證等功能上兩個有所不同,總體來說@validated 相當於 @Valid 驗證的升級版,功能更加強大。

接下來我們直接看下如何使用

引入POM依賴

        <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-validator</artifactId>
            <version>5.2.4.Final</version>
        </dependency>

定義公共分組class(用於標記分組,可以像後面定義在Vo裡面,但是建議一些常用的定義在外部),如下

public interface Add {
}
public interface Edit {
}

定義接收數據的Vo

註意註解中分組的的使用,為瞭演示,同時在內部定義瞭一個特殊分組類

 
import com.example.jsr.commmon.Add;
import com.example.jsr.commmon.Edit;
import org.hibernate.validator.constraints.NotBlank; 
import javax.validation.constraints.Pattern; 
public class ParamsVo {
 
    //特殊用於修改年齡 標記使用 靈活放置位置
    public interface ModifyAge {
    }
 
    //年齡是1-120之間有效
    public static final String AGE_REG = "/^(?:[1-9][0-9]?|1[01][0-9]|120)$/";
 
    @NotBlank(
            groups = {Edit.class, ParamsVo.ModifyAge.class},
            message = "失敗,id不能為空"
    )
    private String id;
 
    @NotBlank(groups = {Edit.class, Add.class}, message = "失敗,名字不能為空")
    private String name;
 
    //自定義一個正則
    @NotBlank(groups = {Add.class, ParamsVo.ModifyAge.class},
            message = "失敗,請填寫age"
    )
    @Pattern(regexp = AGE_REG,groups = {Add.class, ParamsVo.ModifyAge.class},
            message = "失敗,請填寫正確age"
    )
    private String age; 
    ...省略setter getter 方法....
}

統一異常處理類

 
import org.springframework.validation.BindException;
import org.springframework.validation.BindingResult;
import org.springframework.validation.FieldError;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
 
/**
 * 全局異常處理
 */
@ControllerAdvice
public class GlobalExceptionHandler {
 
    @ExceptionHandler(BindException.class)
    @ResponseBody
    public String handlerUnexpectedTypeException(BindException ex){
        BindingResult result = ex.getBindingResult();
        if (result.hasErrors()) {
            FieldError fieldError = result.getFieldError();
            if (fieldError != null) {
                return fieldError.getDefaultMessage();
            }
        }
        return "失敗,請刷新重試";
    } 
    @ExceptionHandler(Exception.class)
    @ResponseBody
    public String handlerException(Exception ex){
        ex.printStackTrace();
        return "失敗,請刷新重試";
    }
}

測試類

import com.example.jsr.Vo.ParamsVo;
import com.example.jsr.commmon.Add;
import com.example.jsr.commmon.Edit;
import org.springframework.stereotype.Controller;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
 
@Controller
@RequestMapping("/validated/test")
public class ValidatedTestController {
 
    @RequestMapping("/add")
    @ResponseBody
    public String add(
            @Validated(Add.class)ParamsVo paramsVo){
        System.out.println(String.format("add obj = {%s}",paramsVo.toString()));
        return "success";
    } 
    @RequestMapping("/edit")
    @ResponseBody
    public String editAll(
            @Validated({Edit.class,ParamsVo.ModifyAge.class})ParamsVo paramsVo){
        System.out.println(String.format("edit obj = {%s}",paramsVo.toString()));
        return "success"; 
    }
}

頁面效果測試

不填age

填入一個錯誤age

到此為止,基本的使用相信也是沒有問題瞭

使用@Validated分組遇到的坑

在使用@Validate註解分組校驗時,如果指定分組,所有的需要驗證的屬性都必須添加指定分組才會校驗

解決辦法

沒有指明分組的屬性都屬於Default,所以分組接口繼承Default就可以解決

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: