Spring利用@Validated注解实现参数校验详解

 更新时间:2023年05月25日 10:12:35   作者:蜀山剑客李沐白  
这篇文章主要为大家详细介绍了在 Spring 项目中使用 @Validated 进行参数校验的方法和常见应用场景,感兴趣的小伙伴可以跟随小编一起学习一下

Spring Framework 提供了一套可以方便地对 Controller 层中接收的参数进行校验的框架,其中就包括了 @Validated 注解。在 Spring 项目中使用 @Validated 注解可以让我们更加方便地进行参数校验,避免了手动校验的麻烦,并且使得代码更加优雅和易于维护。本文将详细介绍在 Spring 项目中使用 @Validated 进行参数校验的方法和常见应用场景。

一、@Validated 注解简介

@Validated 注解是 Spring Framework 中提供的一个参数校验注解,它可以用来标记需要进行参数校验的方法、类、方法参数和方法返回值等地方。通过使用 @Validated 注解,我们可以非常方便地对入参进行检查,并且可以自定义校验规则和错误提示信息。

引入依赖

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

二、@Validated 注解的使用

2.1 在 Controller 层中使用

在 Controller 层中使用 @Validated 注解是最为常见的使用场景。通过在 Controller 方法的参数上添加 @Validated 注解,可以对该参数进行校验。下面是一个简单的例子:

@RestController
@RequestMapping("/api/user")
public class UserController {
    @PostMapping
    public User createUser(@RequestBody @Validated User user) {
        // ...
    }
}

在上述代码中,@Validated 注解标记了 User 类型的参数,表示需要对该参数进行校验。如果 User 类中存在校验注解,那么这些注解会自动触发校验过程。如果校验不通过,则会抛出 MethodArgumentNotValidException 异常。

2.2 自定义校验规则

在使用 @Validated 注解进行参数校验时,我们常常需要自定义校验规则。Spring Framework 提供了多种自定义校验规则的方式,包括使用注解实现、编写自定义 Validator 等方式。

2.2.1 使用注解实现

可以通过编写注解来进行自定义校验规则的实现。例如,下面是一个用于校验手机号码格式的注解:

@Target({ElementType.METHOD, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = PhoneValidator.class)
public @interface Phone {
    String message() default "手机号码格式不正确";
    Class<?>[] groups() default {};
    Class<? extends Payload>[] payload() default {};
}

在上述代码中,@Phone 注解用于标记需要进行手机号码格式校验的字段,而 @Constraint 注解指定了具体的校验逻辑类 PhoneValidator。PhoneValidator 实现了 ConstraintValidator 接口,完成对手机号码格式的校验。

接下来,我们来看一下 PhoneValidator 的实现:

public class PhoneValidator implements ConstraintValidator<Phone, String> {
    private final static Pattern PHONE_PATTERN = Pattern.compile("^1\\d{10}$");
    @Override
    public void initialize(Phone constraintAnnotation) {
    }
    @Override
    public boolean isValid(String phone, ConstraintValidatorContext context) {
        return StringUtils.isEmpty(phone) || PHONE_PATTERN.matcher(phone).matches();
    }
}

在 PhoneValidator 的 isValid 方法中,我们使用了正则表达式来判断手机号码的格式是否正确。如果格式不正确,则返回 false,并且可以通过 context 参数来设置错误提示信息。

使用上述自定义注解实现的校验规则,可以和 Spring 自带的校验注解一样,方便地被应用到 Controller 层中。

2.2.2 编写自定义 Validator

除了使用注解来实现自定义校验规则以外,还可以编写自定义 Validator 来实现具体的校验逻辑。

下面是一个简单的示例,用于校验两个整数的大小关系:

public class ComparisonValidator implements ConstraintValidator<Comparison, Object> {
    private Comparison.Operator operator;
    private String valueFieldName;
    @Override
    public void initialize(Comparison constraintAnnotation) {
        operator = constraintAnnotation.operator();
        valueFieldName = constraintAnnotation.valueFieldName();
    }
    @Override
    public boolean isValid(Object value, ConstraintValidatorContext context) {
        if (value == null) {
            return true;
        }
        try {
            Field valueField = value.getClass().getDeclaredField(valueFieldName);
            valueField.setAccessible(true);
            Object otherValue = valueField.get(value);
            if (otherValue == null) {
                return true;
            }
            int result = ((Comparable) value).compareTo(otherValue);
            switch (operator) {
                case GREATER_THAN:
                    return result > 0;
                case LESS_THAN:
                    return result < 0;
                case GREATER_THAN_OR_EQUAL_TO:
                    return result >= 0;
                case LESS_THAN_OR_EQUAL_TO:
                    return result <= 0;
                default:
                    throw new IllegalArgumentException("Unsupported Comparison Operator: " + operator);
            }
        } catch (NoSuchFieldException | IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    }
}

在上述代码中,我们定义了一个 ComparisonValidator 类,它实现了 ConstraintValidator 接口,并且自定义了一个校验规则 @Comparison。

该校验规则需要通过 operator 和 valueFieldName 两个属性来确定具体的比较方式和被比较的属性名称。在 isValid 方法中,我们首先获取到被比较的属性值 otherValue,然后根据 operator 来判断 value 是否大于(小于、等于)otherValue。

使用自定义 Validator 需要手动创建校验逻辑类,并将其与注解进行关联。在 Controller 层使用时,我们可以像使用 Spring 自带的校验注解一样来使用自定义的校验注解。

2.3 组合注解

有时候,我们需要对同一个参数进行多种校验,这时候可以使用组合注解的方式来实现。例如,下面是一个用于校验密码格式的组合注解示例:

@Target({ElementType.METHOD, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Constraint(validatedBy = {})
@Pattern(regexp = "^(?=.*[0-9])(?=.*[a-z])(?=.*[A-Z])(?=.*[@#$%^&+=])(?=\\S+$).{8,}$", message = "密码必须包含大小写字母、数字和特殊字符,长度至少为 8 位")
@NotNull(message = "密码不能为空")
public @interface Password {
    String message() default "密码格式不正确";
    Class<?>[] groups() default {};
    Class<? extends Payload>[] payload() default {};
}

在上述代码中,@Password 注解通过组合 @Pattern 和 @NotNull 注解来实现了密码的格式校验。如果密码格式不正确或者为空,则会抛出校验异常。
在使用组合注解时,需要注意被组合的注解是否已经使用了 @Constraint 注解,并且不要忘记设置 message、groups 和 payload 等属性。

2.4 统一异常处理

在使用 @Validated 进行参数校验时,如果校验失败,会抛出 MethodArgumentNotValidException 异常。为了提高代码的可维护性,我们可以通过在 Controller 层添加 @ExceptionHandler 注解并捕获该异常,来统一处理校验失败的情况。

例如,下面是一个简单的异常处理示例:

@RestControllerAdvice
public class GlobalExceptionHandler {
    @ExceptionHandler(MethodArgumentNotValidException.class)
    public Map<String, Object> handleValidationException(MethodArgumentNotValidException ex) {
        BindingResult bindingResult = ex.getBindingResult();
        List<String> errorList = bindingResult.getAllErrors().stream()
                .map(DefaultMessageSourceResolvable::getDefaultMessage)
                .collect(Collectors.toList());
        return Collections.singletonMap("message", errorList);
    }
}

在上述代码中,我们通过在 GlobalExceptionHandler 类中添加 @RestControllerAdvice 和 @ExceptionHandler 注解来统一处理 MethodArgumentNotValidException 异常。在 handleValidationException 方法中,我们首先获取到 BindingResult 对象,并通过遍历所有错误信息来收集错误提示信息。最终返回一个包含错误提示信息的 Map 对象。

三、@Validated 注解的常见应用场景

@Validated 注解作为 Spring Framework 中的一个参数校验注解,广泛应用于 Controller 层的参数校验、DTO 类的参数校验和业务类的参数校验等方面。下面列举了几个常见的应用场景:

1.数据库操作时的参数校验:数据库操作一般需要对参数进行校验,避免因为无效参数导致的 SQL 注入等安全问题。

2.DTO 类的参数校验:在使用 DTO(Data Transfer Object)类进行数据传输时,往往需要对传输的字段进行校验,保证数据的有效性和完整性。

3.业务类的参数校验:业务类中的方法通常也需要对参数进行校验,以确保业务逻辑的正确性和可靠性。

四、常用的验证注解

1.@NotNull 注解

@NotNull 表示被注解的参数不能为 null。

例如:

public void testNotNull(@NotNull String str) {}

2.@Size 注解

@Size 表示被注解的参数的大小必须在指定的范围内(包括最小值和最大值)。

例如:

public void testSize(@Size(min = 1, max = 10) String str) {}

3.@Min 和 @Max 注解

@Min 和 @Max 分别表示被注解的参数的最小值和最大值。

例如:

public void testMin(@Min(18) int age) {}
public void testMax(@Max(100) int score) {}

4.@DecimalMin 和 @DecimalMax 注解

@DecimalMin 和 @DecimalMax 分别表示被注解的参数的最小值和最大值,适用于浮点数、BigDecimal 或 BigInteger 类型的参数。

例如:

public void testDecimalMin(@DecimalMin("0.00") BigDecimal price) {}
public void testDecimalMax(@DecimalMax("100.00") BigDecimal score) {}

5.@Digits 注解

@Digits 表示被注解的参数必须是一个数字,并且整数位和小数位的位数不能超过指定的值(默认整数位 2 位,小数位 0 位)。

例如:

public void testDigits(@Digits(integer = 2, fraction = 1) BigDecimal num) {}

6.@Email 注解

@Email 表示被注解的参数必须是一个合法的电子邮件地址。

例如:

public void testEmail(@Email String email) {}

7.@Pattern 注解

@Pattern 表示被注解的参数必须符合指定的正则表达式模式。

例如:

public void testPattern(@Pattern(regexp = "^\\d{4}-\\d{1,2}-\\d{1,2}$") String date) {}

除了使用单个注解外,也可以使用组合注解来完成更为复杂的校验逻辑。

例如:

@NotNull(message = "用户名不能为空")
@Size(min = 5, max = 20, message = "用户名长度必须在 5 到 20 之间")
public String getUsername() {
    return this.username;
}

上述代码表示要求用户名不能为空,并且在长度范围内。如果不符合要求,则会抛出相应的异常,如 MethodArgumentNotValidException 等。

到此这篇关于Spring利用@Validated注解实现参数校验详解的文章就介绍到这了,更多相关Spring Validated内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java微信公众平台开发(13) 微信JSSDK中Config配置

    Java微信公众平台开发(13) 微信JSSDK中Config配置

    这篇文章主要为大家详细介绍了Java微信公众平台开发第十三步,微信JSSDK中Config配置,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-04-04
  • Springmvc自定义异常处理器实现流程解析

    Springmvc自定义异常处理器实现流程解析

    这篇文章主要介绍了Springmvc自定义异常处理器实现流程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-07-07
  • 读取Java文件到byte数组的三种方法(总结)

    读取Java文件到byte数组的三种方法(总结)

    下面小编就为大家带来一篇读取Java文件到byte数组的三种方法(总结)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-08-08
  • 定义hashcode时使用31系数的原因

    定义hashcode时使用31系数的原因

    这篇文章主要介绍了定义hashcode时使用31系数的原因,具有一定借鉴价值,需要的朋友可以参考下
    2018-01-01
  • 解决Mybatis-Plus更新方法不更新NULL字段的问题

    解决Mybatis-Plus更新方法不更新NULL字段的问题

    这篇文章主要介绍了解决Mybatis-Plus更新方法不更新NULL字段的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-12-12
  • idea如何将指定目录打成jar包

    idea如何将指定目录打成jar包

    这篇文章主要介绍了idea如何将指定目录打成jar包问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-09-09
  • JAVA中 redisTemplate 和 jedis的配合使用操作

    JAVA中 redisTemplate 和 jedis的配合使用操作

    这篇文章主要介绍了JAVA中 redisTemplate 和 jedis的配合使用操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-02-02
  • SpringMVC的源码解析

    SpringMVC的源码解析

    本文主要介绍了SpringMVC的源码解析。具有很好的参考价值,下面跟着小编一起来看下吧
    2017-02-02
  • 详解static 和 final 和 static final区别

    详解static 和 final 和 static final区别

    这篇文章主要介绍了static 和 final 和 static final区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-04-04
  • Spring Boot配置元数据方法教程

    Spring Boot配置元数据方法教程

    这篇文章主要介绍了Spring Boot配置元数据方法教程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-12-12

最新评论