当前位置: 首页 > news >正文

Spring Boot Validation实战详解:从入门到自定义规则

目录

一、Spring Boot Validation简介

1.1 什么是spring-boot-starter-validation?

1.2 核心优势

二、快速集成与配置

2.1 添加依赖

2.2 基础配置

三、核心注解详解

3.1 常用校验注解

3.2 嵌套对象校验

四、实战开发步骤

4.1 DTO类定义校验规则

4.2 Controller层启用校验

4.3 统一异常处理

五、高级功能实现

5.1 自定义校验规则

5.2 分组校验

六、常见问题与解决方案

6.1 校验不生效的常见原因

6.2 国际化配置

七、性能优化建议

八、测试验证

8.1 单元测试示例

8.2 API测试(使用MockMvc)

九、总结与最佳实践


一、Spring Boot Validation简介

1.1 什么是spring-boot-starter-validation?

  spring-boot-starter-validation 是Spring Boot对Bean Validation API(JSR 380)的封装实现,基于Hibernate Validator提供强大的数据校验功能。它能帮助开发者:

  • 声明式校验:通过注解定义校验规则

  • 统一错误处理:自动生成标准错误响应

  • 多层级校验:支持DTO、Controller、Service各层

1.2 核心优势

  • 零配置启动:自动装配Validator

  • 丰富注解库:内置30+常用校验规则

  • 高度可扩展:支持自定义校验规则

  • 国际化支持:轻松实现多语言错误提示


二、快速集成与配置

2.1 添加依赖

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

2.2 基础配置

application.yml

spring:messages:basename: i18n/validation # 国际化文件路径encoding: UTF-8server:error:include-message: always # 显示具体错误信息
 

三、核心注解详解

3.1 常用校验注解

注解适用类型说明示例
@NotNull任意类型值不能为null@NotNull(message="ID必填")
@NotEmptyString/Collection非空且长度/大小>0@NotEmpty
@NotBlankString至少包含一个非空格字符@NotBlank
@Size字符串/集合长度/大小范围@Size(min=6, max=20)
@EmailString邮箱格式校验@Email
@PatternString正则表达式匹配@Pattern(regexp="^1[3-9]\\d{9}$")
@Min/@Max数值类型数值范围限制@Min(18)
@Future/@Past时间类型未来/过去时间校验@Future

3.2 嵌套对象校验

public class OrderDTO {@Valid // 启用嵌套校验private UserDTO user;@Validprivate List<@Valid ProductItem> items;
}public class UserDTO {@NotBlankprivate String name;@Emailprivate String email;
}
 

四、实战开发步骤

4.1 DTO类定义校验规则

public class UserCreateRequest {@NotBlank(message = "{user.name.required}")@Size(max = 50, message = "{user.name.length}")private String name;@Email(message = "{user.email.invalid}")private String email;@Pattern(regexp = "^(?=.*[A-Za-z])(?=.*\\d)[A-Za-z\\d]{8,}$", message = "{user.password.policy}")private String password;
}

4.2 Controller层启用校验

@PostMapping("/users")
public ResponseEntity<User> createUser(@RequestBody @Valid UserCreateRequest request) {// 业务逻辑处理return ResponseEntity.ok(userService.create(request));
}

4.3 统一异常处理

@RestControllerAdvice
public class GlobalExceptionHandler {@ExceptionHandler(MethodArgumentNotValidException.class)public ResponseEntity<ErrorResponse> handleValidationException(MethodArgumentNotValidException ex) {List<String> errors = ex.getBindingResult().getFieldErrors().stream().map(error -> error.getField() + ": " + error.getDefaultMessage()).collect(Collectors.toList());return ResponseEntity.badRequest().body(new ErrorResponse("VALIDATION_FAILED", errors));}
}
 

五、高级功能实现

5.1 自定义校验规则

步骤1:创建注解

@Target({FIELD, PARAMETER})
@Retention(RUNTIME)
@Constraint(validatedBy = PhoneNumberValidator.class)
public @interface PhoneNumber {String message() default "{validation.phone.invalid}";Class<?>[] groups() default {};Class<? extends Payload>[] payload() default {};
}

步骤2:实现校验逻辑

public class PhoneNumberValidator implements ConstraintValidator<PhoneNumber, String> {private static final Pattern PHONE_PATTERN = Pattern.compile("^1[3-9]\\d{9}$");@Overridepublic boolean isValid(String value, ConstraintValidatorContext context) {if (value == null) return true; // 允许空值,配合@NotNull使用return PHONE_PATTERN.matcher(value).matches();}
}

5.2 分组校验

public interface CreateGroup {}
public interface UpdateGroup {}public class UserDTO {@Null(groups = CreateGroup.class)@NotNull(groups = UpdateGroup.class)private Long id;@NotBlank(groups = {CreateGroup.class, UpdateGroup.class})private String name;
}@PostMapping("/users")
public ResponseEntity<?> createUser(@RequestBody @Validated(CreateGroup.class) UserDTO dto) {// 创建逻辑
}
 

六、常见问题与解决方案

6.1 校验不生效的常见原因

  1. 缺少@Valid注解:Controller方法参数前忘记添加

  2. 错误异常处理:覆盖了默认的异常处理逻辑

  3. 静态嵌套类:DTO使用static内部类导致无法实例化

  4. 字段访问权限:校验字段需要getter方法

6.2 国际化配置

messages.properties

user.name.required=用户名不能为空
user.email.invalid=邮箱格式不正确
validation.phone.invalid=手机号格式错误

validation_zh_CN.properties

javax.validation.constraints.NotNull.message=不能为null
 

七、性能优化建议

  1. 避免过度校验:只在必要层级进行校验

  2. 合理使用校验组:减少不必要的校验逻辑

  3. 缓存Validator:重复使用Validator实例

@Bean
public Validator validator() {return Validation.buildDefaultValidatorFactory().getValidator();
}
 

八、测试验证

8.1 单元测试示例

@SpringBootTest
public class UserValidationTest {@Autowiredprivate Validator validator;@Testvoid whenInvalidEmail_thenValidationFails() {UserCreateRequest request = new UserCreateRequest();request.setEmail("invalid-email");Set<ConstraintViolation<UserCreateRequest>> violations = validator.validate(request);assertThat(violations).hasSize(1);}
}

8.2 API测试(使用MockMvc)

@WebMvcTest(UserController.class)
class UserControllerTest {@Autowiredprivate MockMvc mvc;@Testvoid createUser_withInvalidPassword_returnsBadRequest() throws Exception {String json = """{"name": "test","email": "test@example.com","password": "123"}""";mvc.perform(post("/users").contentType(APPLICATION_JSON).content(json)).andExpect(status().isBadRequest()).andExpect(jsonPath("$.errors[0]").value("password: 密码必须包含字母和数字"));}
}
 

九、总结与最佳实践

  1. 分层校验原则

    • Controller层:校验输入格式

    • Service层:校验业务规则

    • DAO层:校验数据完整性

  2. 错误消息规范

    • 使用明确的错误代码

    • 保持消息内容用户友好

    • 实现多语言支持

  3. 文档化校验规则

    • 在Swagger文档中展示参数约束

    • 维护校验规则变更日志

相关文章:

  • C++负载均衡远程调用学习之集成测试与自动启动脚本
  • Axure疑难杂症:深度理解与认识“事件”“动作”(玩转交互)
  • 【Linux】Linux入门——权限
  • OpenCV 图形API(81)图像与通道拼接函数-----透视变换函数warpPerspective()
  • 相同的数(简单)
  • 基于 Spring Boot 瑞吉外卖系统开发(十)
  • Spring AI Alibaba-03- Spring AI + DeepSeek-R1 + ES/Milvus + RAG 智能对话应用开发全流程
  • 当手机开始预判你的下一步:一场正在颠覆生活的AI静默革命
  • 解决JSON.stringify方法数据丢失
  • Linux 系统上安装 Firefox 浏览器的完整指南
  • 码蹄集——直线切平面、圆切平面
  • C++入门基础(上)
  • Javase 基础加强 —— 06 Stream流
  • eNSP中路由器OSPF协议配置完整实验和命令解释
  • netty单线程并发量评估对比tomcat
  • Python Bug 修复案例分析:函数参数传递引发的逻辑错误修复
  • C++ 装饰器模式详解
  • 精益数据分析(42/126):移动应用商业模式的深度剖析与实战要点
  • “二次号码焕新”服务来了 可一键解绑历史互联网账号
  • 破解工业3D可视化困局,HOOPS Visualize助力高效跨平台协作与交互!
  • 象屿集团:对去化压力大、市场有效需求不足区域坚决暂停投资,打造多元上市路径
  • 五一假期上海边检查验出入境人员超61万人次,同比增长23%
  • 山大齐鲁医院通报“子宫肌瘤论文现男性患者”:存在学术不端
  • 巴菲特股东大会十大金句:未来五年内可能有投资机会,快乐的人活得更久
  • 长三角铁路今日预计发送旅客398万人次,客流持续保持高位运行
  • 巴菲特批评贸易保护主义:贸易不该被当成武器来使用