服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - spring-boot-starter-validation 校验参数的实现

spring-boot-starter-validation 校验参数的实现

2022-11-11 13:41_lemonV Java教程

参数校验在很多地方都可以用到,本文主要介绍了spring-boot-starter-validation 校验参数的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

一、前言

本章介绍使用spring-boot-starter-validation 校验 SpringMVC 的入参。

	<parent>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-parent</artifactId>
		<version>2.5.2</version>
		<relativePath/> <!-- lookup parent from repository -->
	</parent>
	
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-validation</artifactId>
		</dependency>

 

二、常用注解

spring-boot-starter-validation 校验参数的实现

 

三、定义分组

用于分组校验。
使用场景,对同一个对象例如User(username , id) 在不同的接口时 需要的校验规则不同。
例如,访问一个接口需要 username 不为null且长度大于0 ,id>=0 ; 访问另一个接口 需要 username 参数的长度 在 [1,3]之间。

public class ValidateGroup {
  public interface FirstGroup {
  }

  public interface SecondeGroup {
  }

  public interface ThirdGroup {
  }
}

 

四、定义需要校验的对象

import javax.validation.constraints.Min;
import javax.validation.constraints.NotEmpty;
import javax.validation.constraints.Size;
import lombok.Data;

@Data
public class User {
	@NotEmpty(message = "用户名不能为空")
	@Size(message = "用户名长度 [1-3] ", min = 1, max = 3,groups = ValidateGroup.FirstGroup.class)
	private String username;
	@Min(message = "id不得小于0", value = 0)
	private Integer id;

}

 

五、在handler 即 Controller中 校验

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 com.nbpicc.controller.ValidateGroup.FirstGroup;

@RestController
@RequestMapping("/")
public class TestController {

	@PostMapping("test3")
	public User test3(@RequestBody @Validated({ FirstGroup.class }) User u) {
		System.out.println(u);
		return u;
	}

	@PostMapping("test4")
	public User test4(@Validated User u) {
		System.out.println(u);
		return u;
	}

}

校验失败,会直接抛出异常。这样不太友好,可以使用@ControllerAdvice处理全局异常。

 

六、定义全局异常处理类

import java.util.List;
import org.springframework.validation.BindException;
import org.springframework.validation.BindingResult;
import org.springframework.validation.FieldError;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import lombok.extern.slf4j.Slf4j;

@Slf4j
@RestControllerAdvice
public class GlobalExceptionHandler {

	@ExceptionHandler(value = BindException.class)
	public JsonResult exceptionHandle(BindException exception) {

		BindingResult result = exception.getBindingResult();
		StringBuilder errorMsg = new StringBuilder();

		List<FieldError> fieldErrors = result.getFieldErrors();
		fieldErrors.forEach(error -> {
			log.error("field: " + error.getField() + ", msg:" + error.getDefaultMessage());
			errorMsg.append(error.getDefaultMessage()).append("!");
		});
		return JsonResult.fail(errorMsg.toString());
	}

	@ExceptionHandler(value = MethodArgumentNotValidException.class)
	public JsonResult MyExceptionHandle(MethodArgumentNotValidException exception) {

		BindingResult result = exception.getBindingResult();
		StringBuilder errorMsg = new StringBuilder();

		List<FieldError> fieldErrors = result.getFieldErrors();
		fieldErrors.forEach(error -> {
			log.error("field: " + error.getField() + ", msg:" + error.getDefaultMessage());
			errorMsg.append(error.getDefaultMessage()).append("!");
		});

		return JsonResult.fail(errorMsg.toString());
	}

	// 处理运行时异常
	@ExceptionHandler(RuntimeException.class)
	public JsonResult doHandleRuntimeException(RuntimeException e) {
		log.error(e.getMessage(), e);
		e.printStackTrace();
		return JsonResult.fail(e.getMessage());
	}
}

另外JsonResult.java用于接口返回统一个json格式

import com.fasterxml.jackson.annotation.JsonInclude;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import lombok.experimental.Accessors;

/**
*@author wang 
*@Date 2020-9-14
* 
**/

@Data
@Accessors(chain = true)
@NoArgsConstructor
@AllArgsConstructor 
@JsonInclude(JsonInclude.Include.NON_NULL)
public class JsonResult<T>  {
		/** 成功 */	
		public static final int SUCCESS=200;
		/**内部服务器错误**/
		public static final int FAIL=500;
	    /** 没有登录 */
	    public static final int NOT_LOGIN = 400;
	    /** 发生异常 */
	    public static final int EXCEPTION = 401;
	    /** 系统错误 */
	    public static final int SYS_ERROR = 402;
	    /** 参数错误 */
	    public static final int PARAMS_ERROR = 403;
	    /** 不支持或已经废弃 */
	    public static final int NOT_SUPPORTED = 410;
	    /** AuthCode错误 */
	    public static final int INVALID_AUTHCODE = 444;
	    /** 太频繁的调用 */
	    public static final int TOO_FREQUENT = 445;
	    /** 未知的错误 */
	    public static final int UNKNOWN_ERROR = 499;
		
		private Integer code;  
      private String msg;
      private T data;
      
      public static JsonResult fail() {
          return new JsonResult(FAIL, "请求处理失败",null);
      }
      public static JsonResult fail(String msg) {
          return new JsonResult(FAIL, msg,null);
      }
      public static JsonResult fail(Integer code,String msg) {
          return new JsonResult(code, msg,null);
      }
      public static JsonResult success() {
          return new JsonResult(SUCCESS,"请求处理成功",null);
      }
      public static JsonResult success(String msg) {
          return new JsonResult(SUCCESS,msg,null);
      }
      public static <T> JsonResult success(T data) {
          return new JsonResult<T> (SUCCESS,"请求处理成功",data);
      }
      public static <T>  JsonResult success(String msg,T data) {
          return new JsonResult<T>(SUCCESS, msg,data);
      }
      

      
      public static JsonResult err() {
          return build(EXCEPTION);
      }
      public static JsonResult err(String msg) {
          return build(EXCEPTION, msg);
      }
      
	
      public JsonResult<T> code(int code) {
          this.code = code;
          return this;
      }
      public JsonResult<T> msg(String msg) {
          this.msg = msg;
          return this;
      }
      public JsonResult<T> data(T data) {
          this.data = data;
          return this;
      }
      
      
      public static JsonResult build() {
          return new JsonResult();
      }
      public static JsonResult build(int code) {
          return new JsonResult().code(code);
      }
      public static JsonResult build(int code, String msg) {
          return new JsonResult<String>().code(code).msg(msg);
      }
      public static <T> JsonResult<T> build(int code, T data) {
          return new JsonResult<T>().code(code).data(data);
      }
      public static <T> JsonResult<T> build(int code, String msg, T data) {
          return new JsonResult<T>().code(code).msg(msg).data(data);
      }
      
      
}

当然还有其他异常处理方式可以参考:http://www.tuohang.net/article/226219.html

七、测试效果

spring-boot-starter-validation 校验参数的实现

spring-boot-starter-validation 校验参数的实现

spring-boot-starter-validation 校验参数的实现

 

八、嵌套对象的校验

import lombok.Data;

import javax.validation.Valid;
import javax.validation.constraints.*;

@Data
public class User {
  @NotEmpty(message = "用户名不能为空")
  @Size(message = "长度 [1-3] ", min = 1, max = 3, groups = ValidateGroup.FirstGroup.class)
  private String username;

  @Min(message = "id不得小于0", value = 0)
  private Integer id;
  
  @NotBlank(message = "地址不能为空", groups = {ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class})
  private String address;

  //在内部属性是自定义对象的时候添加 @Valid 注解 ,即可开启对In对象的校验。 
  //记得添加@NotNull注解,否则该对象可以为null,并且此时In对象的校验规则也不会抛出异常 。
  @Valid
  @NotNull(message = "In 对象不能为null ", groups = {ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class})
  private In in;
}
/**
*自定义的对象,User对象中的一个属性。
*/
@Data
public class In {

  @NotBlank(message = "str不能为空", groups = {ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class})
  private String str;
}

 

九、自定义注解(自定义校验规则)

9.1 实现ConstraintValidator接口

import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;

public class WordConstraintValidator implements ConstraintValidator<CustomValidaor, Object> {
	@Override
	public boolean isValid(Object value, ConstraintValidatorContext context) {
		// 具体的校验规则
		return value.toString().length() == 10;
	}
}

9.2 自定义注解

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

import javax.validation.Constraint;
import javax.validation.Payload;

@Target({ ElementType.METHOD, ElementType.FIELD })
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = WordConstraintValidator.class)
public @interface CustomValidaor {

	String message();

	// groups 和 payload 这两个parameter 必须包含,不然会报错
	Class<?>[] groups() default {};

	Class<? extends Payload>[] payload() default {};

}

9.3 使用

@Data
public class Inner {
	@NotBlank(message = "str不能为空", groups = { ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class })
	@CustomValidaor(message = "长度必须为10", groups = { ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class })
	private String str;
}

9.4测试

访问接口

@PostMapping("test6")
public User test6(@Validated({ValidateGroup.SecondeGroup.class}) @RequestBody User u) {
System.out.println(u);
return u;
}

spring-boot-starter-validation 校验参数的实现

到此这篇关于spring-boot-starter-validation 校验参数的实现的文章就介绍到这了,更多相关spring-boot-starter-validation 校验参数内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://www.cnblogs.com/paidaxing7090/p/15074249.html

延伸 · 阅读

精彩推荐
  • Java教程rocketmq如何修改存储路径

    rocketmq如何修改存储路径

    这篇文章主要介绍了rocketmq如何修改存储路径的操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...

    东坡居士爱蹦迪7632021-09-25
  • Java教程JVM 方法调用之动态分派(详解)

    JVM 方法调用之动态分派(详解)

    下面小编就为大家带来一篇JVM 方法调用之动态分派(详解)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    Java教程网1562020-09-22
  • Java教程maven依赖包加载缓慢的原因以及解决方案

    maven依赖包加载缓慢的原因以及解决方案

    这篇文章主要介绍了maven依赖包加载缓慢的原因以及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...

    Java全栈研发大联盟6532022-02-10
  • Java教程Mybatis Generator Plugin悲观锁实现示例

    Mybatis Generator Plugin悲观锁实现示例

    本文将从悲观锁为例,让你快速了解如何实现Mybatis Generator Plugin。文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一...

    raledong9622022-01-20
  • Java教程java几种排序算法的实现及简单分析

    java几种排序算法的实现及简单分析

    这篇文章主要介绍了java几种排序算法的实现及简单分析,实例分析了插入排序、希尔排序、选择排序等常用排序算法,并分析了各个算法的优劣,需要的朋友可...

    hitxueliang4362019-12-19
  • Java教程spring-shiro权限控制realm实战教程

    spring-shiro权限控制realm实战教程

    这篇文章主要介绍了spring-shiro权限控制realm实战教程,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...

    Moshow郑锴5702022-02-21
  • Java教程理解Java中的内存泄露及解决方法示例

    理解Java中的内存泄露及解决方法示例

    这篇文章主要介绍了理解Java中的内存泄露及解决方法示例,本文讲解了Java内存管理机制、Java内存泄露、一般情况下内存泄漏的避免、复杂数据结构中的内存...

    junjie3412019-12-13
  • Java教程java版简单的猜数字游戏实例代码

    java版简单的猜数字游戏实例代码

    猜数字游戏是一款经典的游戏,该游戏说简单也很简单,说不简单确实也很难,那么下面这篇文章主要给大家介绍了java版简单的猜数字游戏的相关资料,文中...

    花花妹子。5232020-10-06