springboot參數校驗


參數校驗

基於PathVariable的正則匹配

通過PathVariable,在{}包裹的字段后加冒號:,在加上正則。
如果不匹配正則,則前端會報出404.

/**
 * 采用正則,在pathvarable中做參數校驗
 * 如果不匹配正則,則返回404
 */
@RestController
@RequestMapping("pathVariable")
public class AppController {

    @GetMapping("/id/{id:\\d+}")
    public String getId(@PathVariable("id") Integer id) {
        return String.valueOf(id);
    }

    @GetMapping("char/{char:[a-zA-Z0-9_]+}")
    public String getChar(@PathVariable("char") String charVar) {
        return charVar;
    }

}

基於注解

添加maven依賴

        <dependency>
            <groupId>javax.validation</groupId>
            <artifactId>validation-api</artifactId>
            <version>2.0.0.Final</version>
        </dependency>
        <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-validator</artifactId>
            <version>6.0.1.Final</version>
        </dependency>

單個字段校驗

1、在Controller上注解@Validated
2、在字段前加入相應注解


/**
 * 單個參數的校驗
 * 在@Validated
 */
@RestController
@RequestMapping("valid1")
@Validated
public class App2Controller {

    @GetMapping
    public String get(@NotEmpty(message = "name不能為空") String name, @Min(value = 12) Integer age) {
        return name + ":" + age;
    }

}

javabean校驗

1、在實體類上加上注解校驗

@Data
public class User {

    @NotEmpty(message = "實體類name不能為空")
    private String name;

    @Max(value = 100)
    private Integer age;
}

2、在Controller出的方法入參處加上@Validated ,BindingResult可以收集錯誤

@RestController
@RequestMapping("user")
public class App3Controller {

    @PostMapping
    public String post(@Validated @RequestBody User user, BindingResult bindingResult) {
        if (bindingResult.hasErrors()) {
            for (ObjectError allError : bindingResult.getAllErrors()) {
                System.out.println(allError.getDefaultMessage());
                throw new RuntimeException(allError.getDefaultMessage());
            }
        }
        return user.getName();
    }
}

自定義注解校驗

1、定義一個注解

@Target({ElementType.ANNOTATION_TYPE, ElementType.METHOD, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = AppValidator.class)//指定注解邏輯實現類
public @interface AppValidAnno {
    String message() default "";

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};
}

note

  • message,groups,payload三個字段必須要
  • 需要在注解上表明邏輯的實現類,這個是ConstraintValidator的實現類
    2、實現ConstraintValidator類
/**
 * 不需要@Component
 * 實現了ConstraintValidator接口,spring會自動將其視為一個組件
 */
//@Component
public class AppValidator implements ConstraintValidator<AppValidAnno, String> {

    //注入其他邏輯類
    @Autowired
    private AppComponent appComponent;

    @Override
    public void initialize(AppValidAnno constraintAnnotation) {
        System.out.println("方法初始化");
    }

    @Override
    public boolean isValid(String s, ConstraintValidatorContext constraintValidatorContext) {
        return appComponent.validStr(s);
    }
}

note

  • 可以在這個類注入其他spring組件
  • 不需要@Component,實現了ConstraintValidator接口,spring會自動將其視為一個組件
  • ConstraintValidator<AppValidAnno, String>兩個范型,第一個是對應的注解,第二個是要處理的字段的類型
    3、和其他校驗注解的使用方式相同
@Data
public class User {

    @NotEmpty(message = "實體類name不能為空")
    private String name;

    @Max(value = 100)
    private Integer age;

    @AppValidAnno(message = "必須以app-開頭")
    private String code;

}

更優雅的方式

采用validation的方式校驗參數的時候,當在參數前加上@valid注解,才會生效,但是在上文的方式中,我們還在代碼中加入了BindingResult這個類來接收錯誤參數。

public String post(@Validated @RequestBody User user, BindingResult bindingResult) {

所以此方法可以簡化這種使用方式,不用在每個方式中加入參數BindingResult。
我們可以采用RestControllerAdvice來統一處理和返回錯誤信息。

    @PostMapping("/o")
    public String post1(@Valid @RequestBody User user) {
        return user.getName();
    }

統一處理代碼

@RestControllerAdvice
public class ExceptionControllerAdvice {
    @ExceptionHandler(MethodArgumentNotValidException.class)
    public String MethodArgumentNotValidExceptionHandler(MethodArgumentNotValidException e) {
        // 從異常對象中拿到ObjectError對象
        ObjectError objectError = e.getBindingResult().getAllErrors().get(0);
        // 然后提取錯誤提示信息進行返回
        return objectError.getDefaultMessage();
    }
}

參考

https://zhuanlan.zhihu.com/p/208298120

git

https://github.com/lexiaoyao1995/validatordemo


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM