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

超詳細講解SpringBoot參數校驗實例

 更新時間:2022年05月13日 14:33:52   作者:Love皮  
經常需要提供接口與用戶交互(獲取數據、上傳數據等),由于這個過程需要用戶進行相關的操作,為了避免出現一些錯誤的數據等,一般需要對數據進行校驗,下面這篇文章主要給大家介紹了關于SpringBoot各種參數校驗的相關資料,需要的朋友可以參考下

使用傳統(tǒng)方式的弊端

public String addUser(User user) {
     if (user == null || user.getId() == null || user.getAccount() == null || user.getPassword() == null || user.getEmail() == null) {
         return "對象或者對象字段不能為空";
     }
     if (StringUtils.isEmpty(user.getAccount()) || StringUtils.isEmpty(user.getPassword()) || StringUtils.isEmpty(user.getEmail())) {
         return "不能輸入空字符串";
     }
     if (user.getAccount().length() < 6 || user.getAccount().length() > 11) {
         return "賬號長度必須是6-11個字符";
     }
     if (user.getPassword().length() < 6 || user.getPassword().length() > 16) {
         return "密碼長度必須是6-16個字符";
     }
     if (!Pattern.matches("^[a-zA-Z0-9_-]+@[a-zA-Z0-9_-]+(\\.[a-zA-Z0-9_-]+)+$", user.getEmail())) {
         return "郵箱格式不正確";
     }
     // 參數校驗完畢后這里就寫上業(yè)務邏輯
     return "success";
 }

這樣做確實沒有什么問題,而且排版也工整,但代碼太繁瑣了,如果有幾十個字段要校驗,那這個方法里面將會變得非常臃腫,實在不夠優(yōu)雅。下面我們就來講講如何使用最優(yōu)雅的方式來解決。

引入依賴

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

注解說明

注解說明
@AssertFalse被注解的元素必須為 false
@AssertTrue被注解的元素必須為 true
@DecimalMax(value)被注解的元素必須是一個數字,其值必須小于等于指定的最大值
@DecimalMin(value)被注解的元素必須是一個數字,其值必須大于等于指定的最小值
@Digits (integer, fraction)被注解的元素必須是一個數字,其值必須在可接受的范圍內
@Null被注解的元素必須為空
@NotNull被注解的元素必須不為空
@Min(value)被注解的元素必須是一個數字,其值必須大于等于指定的最大值
@Max(value)被注解的元素必須是一個數字,其值必須小于等于指定的最大值
@Size(max, min)被注解的元素的長度必須在指定的范圍內
@Past被注解的元素必須是一個過去的日期
@Future被注解的元素必須是一個未來的日期
@Pattern(value)被注解的元素必須符合指定的正則表達式

下面我們以此來在業(yè)務中實現

一、對實體類進行校驗

1、entity

@Data
public class User {
    @NotNull(message = "用戶id不能為空")
    private Long id;
    @NotNull(message = "用戶賬號不能為空")
    @Size(min = 6, max = 11, message = "賬號長度必須是6-11個字符")
    private String account;
    @NotNull(message = "用戶密碼不能為空")
    @Size(min = 6, max = 11, message = "密碼長度必須是6-16個字符")
    private String password;
    @NotNull(message = "用戶郵箱不能為空")
    @Email(message = "郵箱格式不正確")
    private String email;
}

2、controller

@RestController
public class UserController {
    @PostMapping("/addUser")
    public void addUser(@RequestBody @Valid User user) {
		//業(yè)務
    }
}

3、編寫全局統(tǒng)一異常處理

import org.springframework.validation.ObjectError;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import javax.validation.ConstraintViolation;
import javax.validation.ConstraintViolationException;
import java.util.stream.Collectors;
/**
 * 全局異常處理
 *
 * @author master
 */
@RestControllerAdvice
public class ExceptionConfig {
    /**
     * 參數為實體類
     * @param e
     * @return
     */
    @ExceptionHandler(value = MethodArgumentNotValidException.class)
    public String handleValidException(MethodArgumentNotValidException e) {
        // 從異常對象中拿到ObjectError對象
        ObjectError objectError = e.getBindingResult().getAllErrors().get(0);
        // 然后提取錯誤提示信息進行返回
        return objectError.getDefaultMessage();
    }
    /**
     * 參數為單個參數或多個參數
     * @param e
     * @return
     */
    @ExceptionHandler(value = ConstraintViolationException.class)
    public String handleConstraintViolationException(ConstraintViolationException e) {
        // 從異常對象中拿到ObjectError對象
        return e.getConstraintViolations()
                .stream()
                .map(ConstraintViolation::getMessage)
            	.collect(Collectors.toList()).get(0);
    }
}

然后我們使用apipost測試

二、針對單個參數進行校驗

import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.validation.constraints.NotNull;
@RestController
@Validated
public class TestController {
    @GetMapping("/test")
    public void test(@NotNull(message = "id不能為空") Integer id) {
    }
}

然后我們使用apipost測試

三、分組校驗

場景:在新增時我們需要id為空,但修改時我們又需要id不為空,總不可能搞兩個類吧,這時候分組校驗的用處就來了

1、entity

import lombok.Data;
import javax.validation.constraints.Email;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Size;
@Data
public class User {
    public interface Insert{
    }
    public interface Update{
    }
    @NotNull(message = "用戶id不能為空",groups = Update.class)
    @Null(message = "用戶id必須為空",groups = Integer.class)
    private Long id;
    private String account;
    private String password;
    private String email;
}

2、controller

@PostMapping("/add")
public void add(@RequestBody @Validated(User.Insert.class) User user) {
}

添加時就用User.Insert.class,修改時就用User.Update.class

四、自定義分組校驗

場景:當type為1時,需要參數a不為空,當type為2時,需要參數b不為空。

1、entity

import com.example.demo.provider.CustomSequenceProvider;
import lombok.Data;
import org.hibernate.validator.group.GroupSequenceProvider;
import javax.validation.constraints.NotEmpty;
import javax.validation.constraints.Pattern;
@Data
@GroupSequenceProvider(value = CustomSequenceProvider.class)
public class CustomGroup {
    /**
     * 類型
     */
    @Pattern(regexp = "[A|B]" , message = "類型不必須為 A|B")
    private String type;
    /**
     * 參數A
     */
    @NotEmpty(message = "參數A不能為空" , groups = {WhenTypeIsA.class})
    private String paramA;
    /**
     * 參數B
     */
    @NotEmpty(message = "參數B不能為空", groups = {WhenTypeIsB.class})
    private String paramB;
    /**
     * 分組A
     */
    public interface WhenTypeIsA {
    }
    /**
     * 分組B
     */
    public interface WhenTypeIsB {
    }
}

2、CustomSequenceProvider

import com.example.demo.controller.CustomGroup;
import org.hibernate.validator.spi.group.DefaultGroupSequenceProvider;
import java.util.ArrayList;
import java.util.List;
public class CustomSequenceProvider implements DefaultGroupSequenceProvider<CustomGroup> {
    @Override
    public List<Class<?>> getValidationGroups(CustomGroup form) {
        List<Class<?>> defaultGroupSequence = new ArrayList<>();
        defaultGroupSequence.add(CustomGroup.class);
        if (form != null && "A".equals(form.getType())) {
            defaultGroupSequence.add(CustomGroup.WhenTypeIsA.class);
        }
        if (form != null && "B".equals(form.getType())) {
            defaultGroupSequence.add(CustomGroup.WhenTypeIsB.class);
        }
        return defaultGroupSequence;
    }
}

3、controller

@PostMapping("/add")
public void add(@RequestBody @Validated CustomGroup user) {
}

五、自定義校驗

雖然官方提供的校驗注解已經滿足很多情況了,但還是無法滿足我們業(yè)務的所有需求,比如校驗手機號碼,下面我就以校驗手機號碼來做一個示例。

1、定義校驗注解

@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = PhoneValidator.class)
public @interface Phone {
    String message() default "手機號碼格式有誤";
    Class<?>[] groups() default {};
    Class<? extends Payload>[] payload() default {};
}

注:groups和payload是必須要寫的,Constraint是使用哪個類來進行校驗。

2、實現注解

import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;
import java.util.regex.Pattern;
/**
 * @author master
 */
public class PhoneValidator implements ConstraintValidator<Phone, Object> {
    @Override
    public boolean isValid(Object telephone, ConstraintValidatorContext constraintValidatorContext) {
        String pattern = "^1[3|4|5|6|7|8|9]\\d{9}$";
        return Pattern.matches(pattern, telephone.toString());
    }
}

最后直接用到參數前面或者實體類變量上面即可。

六、嵌套校驗

當某個對象中還包含了對象需要進行校驗,這個時候我們需要用嵌套校驗。

@Data
public class TestAA {
    @NotEmpty(message = "id不能為空")
    private String id;
    @NotNull
    @Valid
    private Job job;
    @Data
    public class Job {
        @NotEmpty(message = "content不能為空")
        private String content;
    }
}

七、快速失敗

Spring Validation默認會校驗完所有字段,然后才拋出異常??梢酝ㄟ^配置,開啟Fali Fast模式,一旦校驗失敗就立即返回。

import org.hibernate.validator.HibernateValidator;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import javax.validation.Validation;
import javax.validation.Validator;
import javax.validation.ValidatorFactory;
@Configuration
public class FailFastConfig {
    @Bean
    public Validator validator() {
        ValidatorFactory validatorFactory = Validation.byProvider(HibernateValidator.class)
                .configure()
                // 快速失敗模式
                .failFast(true)
                .buildValidatorFactory();
        return validatorFactory.getValidator();
    }
}

注意事項

SpringBoot 2.3.x 移除了validation依賴需要手動引入依賴。

總結

非空校驗是校驗的第一步, 除了非空校驗,我們還需要做到以下幾點:

  • 普通參數 - 需要限定字段的長度。如果會將數據存入數據庫,長度以數據庫為準,反之根據業(yè)務確定。
  • 類型參數 - 最好使用正則對可能出現的類型做到嚴格校驗。比如type的值是【0|1|2】這樣的。
  • 列表(list)參數 - 不僅需要對list內的參數是否合格進行校驗,還需要對list的size進行限制。比如說 100。
  • 日期,郵件,金額,URL這類參數都需要使用對于的正則進行校驗。
  • 參數真實性 - 這個主要針對于 各種Id 比如說 userId、merchantId,對于這樣的參數,都需要進行真實性校驗

參數校驗越嚴格越好,嚴格的校驗規(guī)則不僅能減少接口出錯的概率,同時還能避免出現臟數據,從而來保證系統(tǒng)的安全性和穩(wěn)定性。

到此這篇關于超詳細講解SpringBoot參數校驗實例的文章就介紹到這了,更多相關SpringBoot參數校驗內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

最新評論