SpringBoot参数校验(javax.validation)

本文介绍了一种使用 javax.validation 进行前后端数据交互参数校验的方法,通过实例展示了如何封装请求参数类并利用注解简化校验过程。

问题描述

我们在进行前后端数据交互的时候,总要进行数据非空校验以及其他的一些相关的操作,我们可以用
javax.validation.Validation包下的注解进行验证,实现简单

给大家看一下我刚工作时候写的参数校验OvO

public String studentLeaveView(@RequestHeader("schoolId") String schoolId, String leaveId, Integer type, String taskId) {
        Map<String, Object> map = new HashMap<>();
        if (StringUtils.isEmpty(schoolId)) {
            return ApiResponseBase.result(ApiResponseCode.SCHOOL_ID_IS_NULL, "学校id为空");
        } else if (StringUtils.isEmpty(leaveId)) {
            return ApiResponseBase.result(ApiResponseCode.LEAVE_ID_IS_NULL, "假条id为空");
        } else if (StringUtils.isNull(type)) {
            return ApiResponseBase.result(ApiResponseCode.LEAVE_TYPE_IS_NULL, "请假类型为空");
        }
        //业务处理
       }

这个接口只有三个参数需要检验,所以现在看起来不是特别繁琐,但是如果参数多呢,试想一下是不是惨不忍睹。。。。

利用javax.validation解决

  1. 编写实体类
package com.komlin.project.api.dto;

import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;

import javax.validation.constraints.NotNull;

/**
 * @Auther: mt
 * @Date: 2020/08/03 11:00
 * @Contact: 1692314230@qq.com
 * @Description: 请求参数类封装
 */
@Data
@Builder
@AllArgsConstructor
@NoArgsConstructor
public class TestDTO {

	//非空注解
    @NotNull(message = "param1 is not null")
    private String param1;


    @NotNull(message = "param2 is not null")
    private String param2;


}

  1. 进行校验
package com.komlin.project.api.controller;

import com.komlin.common.enums.ResponseEnum;
import com.komlin.common.pojo.ResponseEntity;
import com.komlin.common.utils.ValidationUtils;
import com.komlin.project.api.dto.TestDTO;
import com.komlin.project.api.vo.TestVO;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.validation.BindingResult;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.validation.Valid;

import static com.komlin.common.constant.HttpMethodConstant.POST;

/**
 * @Auther: mt
 * @Date: 2020/08/03 11:00
 * @Contact: 1692314230@qq.com
 * @Description: Demo
 */
@Api(tags = "参考模板")
@RestController
@Slf4j
@RequestMapping("api/test")
public class ApiExampleController {

    @PostMapping("/")
    @ApiOperation(value = "post测试",httpMethod = POST)
    public ResponseEntity<TestVO> postTest(@RequestBody @Valid TestDTO test, BindingResult bindingResult){

        //校验参数
        if (bindingResult.hasErrors()) {
            return ValidationUtils.validate(bindingResult);
        }
        //正确返回
//        return ResponseEntity.ok();
//        return ResponseEntity.ok(new TestVO());


        //错误返回
        return ResponseEntity.error(ResponseEnum.SERVER_EXCEPTION);
    }
}

ValidationUtils

package com.komlin.common.utils;

import com.gexin.fastjson.JSON;
import com.gexin.fastjson.JSONObject;
import com.komlin.common.pojo.ErrorResult;
import com.komlin.common.pojo.ResponseBase;
import com.komlin.common.pojo.ResponseEntity;
import com.komlin.project.common.response.ApiResponseBase;
import org.springframework.validation.BindingResult;

import java.util.ArrayList;
import java.util.List;

/**
 * @Auther: mt
 * @Date: 2020-08-01 12:23
 * @Description: 信息校验
 */
public class ValidationUtils {
    /**
     * 校验请求参数
     *
     * @param bindingResult
     * @return
     */
    public static ResponseBase validateParams(BindingResult bindingResult) {
        List<ErrorResult> errorResultList = new ArrayList<>();
        if (bindingResult.hasErrors()) {
            bindingResult.getFieldErrors().forEach(fieldError -> {
                errorResultList.add(ErrorResult.builder()
                        .param(fieldError.getField())
                        .message(fieldError.getDefaultMessage())
                        .build());
            });
        }
        return ResponseBase.paramError("Incorrect request parameter", errorResultList);
    }


    /**
     * 校验请求参数
     *
     * @param bindingResult
     * @return
     */
    public static ResponseEntity validate(BindingResult bindingResult) {
        List<ErrorResult> errorResultList = new ArrayList<>();
        if (bindingResult.hasErrors()) {
            bindingResult.getFieldErrors().forEach(fieldError -> {
                errorResultList.add(ErrorResult.builder()
                        .param(fieldError.getField())
                        .message(fieldError.getDefaultMessage())
                        .build());
            });
        }
        return new ResponseEntity(400, JSON.toJSONString(errorResultList), null);
    }

    /**
     * 校验请求参数
     * @param bindingResult
     * @return
     */
    public static String validateParam(BindingResult bindingResult) {
        List<ErrorResult> errorResults = new ArrayList<>();
        if (bindingResult.hasErrors()) {
            bindingResult.getFieldErrors().forEach(fieldError -> {
                errorResults.add(ErrorResult.builder()
                        .param(fieldError.getField())
                        .message(fieldError.getDefaultMessage())
                        .build());

            });
        }
        return ApiResponseBase.result(3000,JSONObject.toJSONString(errorResults.get(0).getMessage()));
    }
}

附录:注解说明

注解支持的数据类型作用
@AssertFalseBoolean,boolean值等于false
@AssertTrueBoolean,boolean值等于true
@Futurejava.util.Date等检查给定的日期比当前日期晚
@Pastjava.util.Date等检查给定的日期比当前日期早
@MaxBigDecimal, BigInteger, byte, short, int, long 以及包装类检查该值是否小于或等于约束条件中指定的最大值.
@MinBigDecimal, BigInteger, byte, short, int, long 以及包装类检查该值是否大于或等于约束条件中指定的最小值.
@NotNull任意类型带注释的值不为null.
@Null任意类型带注释的值为null.
@Pattern(regex=, flag=)String检查该字符串是否能够在match指定的情况下被regex定义的正则表达式匹配.
@Size(min=, max=)String,Collection,Map,array检查带注释的元素的大小是否在最小值和最大值之间.
在验证时间戳格式时,`javax.validation.constraints` 包中并没有直接支持毫秒级时间戳的注解。常用的日期和时间验证注解包括 `@Future`、`@Past`、`@FutureOrPresent` 和 `@PastOrPresent`,这些注解可以用于 `java.time.Instant`、`java.util.Date` 等类型来验证时间戳是否符合预期的时间关系[^1]。 ### 验证时间戳的方法 #### 1. 使用 `@Future` 和 `@Past` - **`@Future`**:确保时间戳在未来。 - **`@Past`**:确保时间戳在过去。 - **`@FutureOrPresent`**:确保时间戳在未来或当前时间。 - **`@PastOrPresent`**:确保时间戳在过去或当前时间。 示例代码: ```java import javax.validation.constraints.Future; import java.time.Instant; public class TimestampExample { @Future(message = "时间戳必须在未来") private Instant futureTimestamp; // Getter and Setter } ``` #### 2. 自定义注解验证时间戳格式 如果需要验证特定格式的时间戳(例如毫秒级时间戳),可以通过自定义注解和校验器实现。 ##### 步骤: 1. 定义自定义注解 `@ValidTimestamp`。 2. 实现对应的校验器类 `ValidTimestampValidator`。 示例代码: ```java import javax.validation.Constraint; import javax.validation.Payload; import java.lang.annotation.*; @Documented @Constraint(validatedBy = ValidTimestampValidator.class) @Target({ ElementType.METHOD, ElementType.FIELD }) @Retention(RetentionPolicy.RUNTIME) public @interface ValidTimestamp { String message() default "无效的时间戳格式"; Class<?>[] groups() default {}; Class<? extends Payload>[] payload() default {}; } ``` ```java import javax.validation.ConstraintValidator; import javax.validation.ConstraintValidatorContext; public class ValidTimestampValidator implements ConstraintValidator<ValidTimestamp, Long> { @Override public void initialize(ValidTimestamp constraintAnnotation) { } @Override public boolean isValid(Long value, ConstraintValidatorContext context) { if (value == null) { return true; // 允许 null 值 } try { // 检查是否为有效的毫秒级时间戳 return value >= 0 && value <= System.currentTimeMillis() + 1000L * 60 * 60 * 24 * 365 * 10; // 10年以内 } catch (Exception e) { return false; } } } ``` 使用自定义注解: ```java public class TimestampExample { @ValidTimestamp(message = "时间戳格式不正确") private Long timestamp; // Getter and Setter } ``` #### 3. 配置 Spring Boot 验证 确保在 Spring Boot 中启用了验证功能,在配置类或主应用类上添加 `@EnableValidation` 注解。 ```java import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.context.annotation.EnableValidation; @SpringBootApplication @EnableValidation public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } } ``` ### 异常处理 为了更好地返回错误信息给前端,可以在全局异常处理器中捕获 `ConstraintViolationException` 并返回友好的错误消息[^3]。 示例代码: ```java import org.springframework.web.bind.annotation.ExceptionHandler; import org.springframework.web.bind.annotation.RestControllerAdvice; import javax.validation.ConstraintViolationException; import java.util.HashMap; import java.util.Map; @RestControllerAdvice public class ValidationExceptionHandler { @ExceptionHandler(ConstraintViolationException.class) public Map<String, String> handleValidationExceptions(ConstraintViolationException ex) { Map<String, String> errors = new HashMap<>(); ex.getConstraintViolations().forEach(violation -> { String field = violation.getPropertyPath().toString(); String message = violation.getMessage(); errors.put(field, message); }); return errors; } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值