目錄
使用传统方式的弊端
引入依赖
注解说明
一、对实体类进行校验
1、entity
3、controller
3、编写全局统一异常处理
二、针对单个参数进行校验
三、分组校验
四、自定义分组校验
2、CustomSequenceProvider
五、自定义校验
1、定义校验注解
2、实现注解
六、嵌套校验
七、快速失败
注意事项
首頁 Java java教程 SpringBoot參數怎麼校驗

SpringBoot參數怎麼校驗

May 14, 2023 pm 10:04 PM
springboot

    使用传统方式的弊端

    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 "邮箱格式不正确";
         }
         // 参数校验完毕后这里就写上业务逻辑
         return "success";
     }
    登入後複製

    这样做确实没有什么问题,而且排版也工整,但代码太繁琐了,如果有几十个字段要校验,那这个方法里面将会变得非常臃肿,实在不够优雅。下面我们就来讲讲如何使用最优雅的方式来解决。

    引入依赖

    <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)被注解的元素必须符合指定的正则表达式

    下面我们以此来在业务中实现

    一、对实体类进行校验

    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) {
    		//业务
        }
    }
    登入後複製

    3、编写全局统一异常处理

    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测试

    SpringBoot參數怎麼校驗

    二、针对单个参数进行校验

    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测试

    SpringBoot參數怎麼校驗

    三、分组校验

    场景:在新增时我们需要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) {
    }
    登入後複製

    五、自定义校验

    虽然官方提供的校验注解已经满足很多情况了,但还是无法满足我们业务的所有需求,比如校验手机号码,下面我就以校验手机号码来做一个示例。

    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依赖需要手动引入依赖。

    以上是SpringBoot參數怎麼校驗的詳細內容。更多資訊請關注PHP中文網其他相關文章!

    本網站聲明
    本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

    熱AI工具

    Undresser.AI Undress

    Undresser.AI Undress

    人工智慧驅動的應用程序,用於創建逼真的裸體照片

    AI Clothes Remover

    AI Clothes Remover

    用於從照片中去除衣服的線上人工智慧工具。

    Undress AI Tool

    Undress AI Tool

    免費脫衣圖片

    Clothoff.io

    Clothoff.io

    AI脫衣器

    AI Hentai Generator

    AI Hentai Generator

    免費產生 AI 無盡。

    熱門文章

    R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
    2 週前 By 尊渡假赌尊渡假赌尊渡假赌
    倉庫:如何復興隊友
    4 週前 By 尊渡假赌尊渡假赌尊渡假赌
    Hello Kitty Island冒險:如何獲得巨型種子
    3 週前 By 尊渡假赌尊渡假赌尊渡假赌

    熱工具

    記事本++7.3.1

    記事本++7.3.1

    好用且免費的程式碼編輯器

    SublimeText3漢化版

    SublimeText3漢化版

    中文版,非常好用

    禪工作室 13.0.1

    禪工作室 13.0.1

    強大的PHP整合開發環境

    Dreamweaver CS6

    Dreamweaver CS6

    視覺化網頁開發工具

    SublimeText3 Mac版

    SublimeText3 Mac版

    神級程式碼編輯軟體(SublimeText3)

    Springboot怎麼整合Jasypt實現設定檔加密 Springboot怎麼整合Jasypt實現設定檔加密 Jun 01, 2023 am 08:55 AM

    Springboot怎麼整合Jasypt實現設定檔加密

    怎麼在SpringBoot中使用Redis實現分散式鎖 怎麼在SpringBoot中使用Redis實現分散式鎖 Jun 03, 2023 am 08:16 AM

    怎麼在SpringBoot中使用Redis實現分散式鎖

    SpringBoot怎麼整合Redisson實現延遲隊列 SpringBoot怎麼整合Redisson實現延遲隊列 May 30, 2023 pm 02:40 PM

    SpringBoot怎麼整合Redisson實現延遲隊列

    SpringBoot與SpringMVC的比較及差別分析 SpringBoot與SpringMVC的比較及差別分析 Dec 29, 2023 am 11:02 AM

    SpringBoot與SpringMVC的比較及差別分析

    springboot讀取檔案打成jar包後存取不到怎麼解決 springboot讀取檔案打成jar包後存取不到怎麼解決 Jun 03, 2023 pm 04:38 PM

    springboot讀取檔案打成jar包後存取不到怎麼解決

    SpringBoot怎麼自訂Redis實作快取序列化 SpringBoot怎麼自訂Redis實作快取序列化 Jun 03, 2023 am 11:32 AM

    SpringBoot怎麼自訂Redis實作快取序列化

    Springboot+Mybatis-plus不使用SQL語句進行多表新增怎麼實現 Springboot+Mybatis-plus不使用SQL語句進行多表新增怎麼實現 Jun 02, 2023 am 11:07 AM

    Springboot+Mybatis-plus不使用SQL語句進行多表新增怎麼實現

    springboot怎麼取得application.yml裡值 springboot怎麼取得application.yml裡值 Jun 03, 2023 pm 06:43 PM

    springboot怎麼取得application.yml裡值

    See all articles