Java数据校验详解&springboot 统一异常处理(包含统一数据校验)

一切从元编程开始

一个健壮的系统都要对外部提交的数据进行完整性、合法性的校验。即使开发一个不面对最终用户的工具包,也需要对传入的数据进行缜密的校验来防止引发底层难以追踪的问题。各路大神当然也会注意到这个问题,所以在“元编程”(见JSR250与资源控制)提出之后相续提交了JSR-303、JSR-349以及JSR-380来完善使用注解进行数据校验的机制,这三个JSR也被称为Bean Validation 1.0、Bean Validation 1.1和Bean Validation 2.0,后文统称为Bean Validation。

先看一个不使用Bean Validation校验数据的代码:

public class StandardValidation {

	public static void main(String[] args) {
		System.out.println(validationWithoutAnnotation(" ", -1));
	}

	public static String validationWithoutAnnotation(String inputString, Integer inputInt) {
		String error = null;
		if (null == inputString) {
			error = "inputString不能为null";
		} else if (null == inputInt) {
			error = "inputInt不能为null";
		} else if (1 > inputInt.compareTo(0)) {
			error = "inputInt必须大于0";
		} else if (inputString.isEmpty() || inputString.trim().isEmpty()) {
			error = "inputString不能为空字符串";
		} else {
			// DO
		}
		return error;
	}
}

相信很多码友多少都写过类似的代码。使用IF—ELSE是否优雅这种高端问题暂且不谈,但是大量的IF—ELSE会导致业务内容越来越多的嵌套在代码中。针对这些问题Bean Validation为数据校验提供了更加规范化、通用化、复用程度更高的校验方法。

数据校验的原理并不复杂,主要是用注解(Annotation)在域或setter方法上声明JavaBean中数据的准则。Java的数据校验代码主要在javax.validation包中,包括注解、校验器以及校验器工厂,接下来通过例子说明。(例子可执行代码在本人的gitee库,本文代码在chkui.springcore.example.javabase.validation包)

标准数据校验

JSR提交的Javax.validation定义中已经为数据校验定义了很多方法和注解,但是需要清晰的是JSR仅仅制定了一个规范,具体的功能是由各种框架实现的。本文的例子引入了Hibernate Validator 6.0.12.Final包,他与Spring Validator一样,都是根据JSR规范实现校验功能。

数据校验是围绕一个实体类展开的,下面的代码声明了一个实体类,通过注解标注每个域上的赋值规则:

package chkui.springcore.example.javabase.validation.entity;
public class Game {
	@NotNull //非空
	@Length(min=0, max=5) //字符串长度小于5,这个是一个Hibernate Validator增加的注解
	private String name;
	
	@NotNull
	private String description;
	
	@NotNull
	@Min(0) //最小值>=0
	@Max(10) //最大值<=10
	private int currentVersion; 
    //getter and setter…………
}

使用校验器对其进行校验:

public StandardValidation {
	public void validate() {
		//引入校验工具
		ValidatorFactory factory = Validation.buildDefaultValidatorFactory();
        //获取校验器
		Validator validator = factory.getValidator();
		Game wow = new Game();
        //执行校验
		Set> violationSet = validator.validate(wow);
		violationSet.forEach(violat -> {
			violat.getPropertyPath();//校验错误的域
            violat.getMessage());//校验错误的信息
		});
        //设置值之后再次进行校验
		wow.setName("World Of Warcraft");
		wow.setDescription("由著名游戏公司暴雪娱乐所制作的第一款网络游戏,属于大型多人在线角色扮演游戏。");
		wow.setCurrentVersion(8);
		violationSet = validator.validate(wow);
		violationSet.forEach(violat -> {});
	}
}

执行完毕之后violationSet中就是校验的结果。如果校验通过那么返回的Set长度为0。

Bean Validation已经为常规的校验功能预设了很多注解,详见关于所有注解的介绍。

自定义校验规则

虽然在javax.validation.constraints已经定义了很多用于校验的注解,但是肯定无法满足复杂多样的业务需求。所以Bean Validation也支持自定义校验规则。在JSR的文档中对数据域的一个校验被称为Constraint(约束),一个Constraint由一个Annotation(注解)绑定1~n个Validator(校验器)组成。 因此可以通过新增AnnotationValidator来定义新的校验方式(或者说是定义新的Constraint)。

组合注解校验

可以通过组合已有的注解来实现新的数据校验规则。例如下面的例子。

定义新的校验注解:

package chkui.springcore.example.javabase.validation.annotation;
@Min(1)//最小值>=1
@Max(300)//最大值<=300
@Constraint(validatedBy = {}) //不制定校验器
@Documented
@Target({ ElementType.ANNOTATION_TYPE, ElementType.METHOD, ElementType.FIELD })
@Retention(RetentionPolicy.RUNTIME)
public @interface Price {
	String message() default "定价必须在$1~$200之间";
	Class[] groups() default { };
	Class[] payload() default { };
}

在@Price注解中我们标记了@Min(1)和@Max(300),之后直接在域上标记@Price就会校验对应的值是否满足这个条件:

package chkui.springcore.example.javabase.validation.entity;
public class Game {
    @Price
	private float price;
    //Other field
    //setter and getter
}

自定义校验器

除了组合javax.validation.constraints中的注解,还可以自定义校验器(Validator)进行数据校验。

声明一个用于自定义校验的注解:

package chkui.springcore.example.javabase.validation.annotation;
@Constraint(validatedBy = { TypeValidator.class }) //指定校验器
@Documented
@Target({ ElementType.ANNOTATION_TYPE, ElementType.METHOD, ElementType.FIELD })
@Retention(RetentionPolicy.RUNTIME)
public @interface Type {
	String message() default "游戏类型错误,可选类型为RPG、ACT、SLG、ARPG";
	Class[] groups() default {};
	Class[] payload() default {};
}

注意@Constraint(validatedBy = { TypeValidator.class })这一行代码,他的作用就是将这个注解和校验器进行绑定,当我们执行Validator::validator方法时对应的校验器会被调用。

TypeValidator类:

package chkui.springcore.example.javabase.validation.validator;
public class TypeValidator implements ConstraintValidator {
	private final List TYPE = Arrays.asList(new String[]{"RPG", "ACT", "SLG", "ARPG"});
	@Override
	public boolean isValid(String value, ConstraintValidatorContext context) {
		return TYPE.contains(value);
	}
}

TypeValidator必须实现ConstraintValidator这个接口,并在范型中声明对应的校验注解和数据类型(ConstraintValidator,T是绑定的注解类型、E是数据类型)。TypeValidator中判断数值是不是"RPG", "ACT", "SLG", "ARPG"当中的一个,若不是则TypeValidator::isValid返回false表示校验没通过。

在实体类的域上使用自定义的@Type注解:

public class Game {
	@NotNull
	@Type
	private String type;
    //Other field ......
    //getter and setter ......
}

分组校验

对于业务来说数据录入的规则并不是一成不变的,往往需要根据某些状态来对单个或一组数据进行校验。这个时候我们可以用到分组功能——根据状态启用一组约束。

观察自定义注解或javax.validation.constraints包中预定以的注解,都有一个groups参数:

public @interface Max {
	String message() default "{javax.validation.constraints.Max.message}";
	Class[] groups() default { }; //用于分组的参数
	Class[] payload() default { };
	long value();
}

如果未指定该参数,那么校验都属于javax.validation.groups.Default分组。

先定义一个分组,用一个没有任何功能的类或者接口即可:

package chkui.springcore.example.javabase.validation.groups;
public interface BetaGroup {}

然后在校验的注解上通过groups指定分组:

public class Game {
	
	@NotNull
	@Min(0) //最小值>=0
	@Max(10) //最大值<=10
	@Max(value=0, message="未发行的游戏版本为0!", groups = BetaGroup.class)//分组校验
	private int currentVersion; 
	
	@AssertTrue(groups = BetaGroup.class)//分组校验
	//表示是否为内侧版
	private boolean beta;
    //Other field ......
    //getter and setter ......
}

然后执行分组校验:

public enum StandardValidation {
	public void validate() {
		//引入校验工具
		ValidatorFactory factory = Validation.buildDefaultValidatorFactory();
		Validator validator = factory.getValidator();

		Game wow = new Game();
		wow.setName("World Of Warcraft");
		wow.setDescription("由著名游戏公司暴雪娱乐所制作的第一款网络游戏,属于大型多人在线角色扮演游戏。");
		wow.setCurrentVersion(8);
		wow.setType("RPG");
		wow.setPrice(401.01F);

        //使用默认分组校验
		violationSet = validator.validate(wow);
		
		//指定分组校验
		violationSet = validator.validate(wow, BetaGroup.class);
	}
}

Validator::validator方法未指定分组时,相当于使用javax.validation.groups.Default分组。而在violationSet=validator.validate(wow, BetaGroup.class);这一行代码指定分组之后,只会执行groups = BetaGroup.class注解的校验。

可以一次指定多个分组的校验,这样有利于处理复杂的状态:

validator.validate(wow, Default.class, BetaGroup.class, OtherGroup.class);

校验错误级别

校验的注解中还有一个参数——payload,他表示“校验问题”的级别。这个参数就像使用Log4j输出日志会指定DEBUG、INFO、WARN等级别一样,在校验数据时会有对“校验问题”进行分类的需求,比如某些页面会对用户录入的数据进行“错误”或“警告”的提示。

在使用payload时需要先声明PalyLoad接口类以标定“问题级别”:

package chkui.springcore.example.javabase.validation;
public class PayLoadLevel {
    //警告级别
	static public interface WARN extends Payload {}
    //错误级别
	static public interface Error extends Payload {}
}

然后在JavaBean上指定“校验问题”的级别:

public class Game {
    //默认分组校验错误时,错误级别为Error
	@NotNull(payload=PayLoadLevel.Error.class)
	@Min(value=0, payload=PayLoadLevel.Error.class) 
	@Max(value=10, payload=PayLoadLevel.Error.class) 
    //BetaGroup分组错误级别为WARN
	@Max(value=0, message="未发行的游戏版本为0!", groups = BetaGroup.class, payload=PayLoadLevel.WARN.class)
	private int currentVersion; 
	
	@AssertTrue(groups = BetaGroup.class, payload=PayLoadLevel.WARN.class)
	private boolean beta;
    //Other field ......
    //getter and setter ......	
}

然后在执行校验的时候使用ConstraintViolation::getConstraintDescriptor::getPayload方法获取每一个校验问题的错误级别:

violationSet = validator.validate(wow, BetaGroup.class);
violationSet.forEach(violat -> {
	violat.getPropertyPath();//错误域的名称
    violat.getMessage();//错误消息
	violat.getConstraintDescriptor().getPayload();//错误级别
});

springboot 统一异常处理(包含统一数据校验)

1、统一异常处理的优势

在开发中,我们是否遇到过如下两种奇葩现象:

(1)只要没有成功,不管什么原因,前端界面给出提示:服务端错误/异常。哪怕是数据校验不过,也这样提示(嗯,反正先把锅甩出去再说,具体什么原因我才不在乎呢,老子就是这么聪明);

如果想学习Java工程化、高性能及分布式、深入浅出。微服务、Spring,MyBatis,Netty源码分析的朋友可以加我的Java高级交流:787707172,群里有阿里大牛直播讲解技术,以及Java大型互联网技术的视频免费分享给大家。

(2)前端不做任何提示,一切提示信息都来自后端,成功的时候自然没什么,失败的时候,比如将Exception的描述信息(e.getMessage)返回。

现象(1)没什么好说的,直接拖出去枪毙吧;现象(2)先把产品经理宰了再说吧,看起来好像很专业的样子,出了什么问题直接看response返回的结果就知道个大概,研发测试都很方便,只是,大家想过没有,研发测试运维的问题,凭什么要用户买单,你见过淘宝京东有时候出了问题给你类似于“out of memory”的异常提示吗?

那么异常统一处理有什么好处呢?

提高用户体验;

业务逻辑和异常处理逻辑解耦;

对异常进行分类统一处理,减少冗余代码;

便于代码风格统一,并且更优雅(比如参数校验的时候,得写很多if else,并且不同的人写法不一致);

2、统一异常处理的实现

2.1 springboot的默认异常处理

Spring Boot提供了一个默认的映射:/error,当处理中抛出异常之后,会转到该请求中处理,并且该请求有一个全局的错误页面用来展示异常内容。

比如:

@RestController
public class Test {
 @RequestMapping(value = {"/test"},method = RequestMethod.GET)
 public String test(@RequestParam("id")Integer id){
 return "id:"+id;
 }
}

运行后访问结果如下:

Java数据校验详解&springboot 统一异常处理(包含统一数据校验)_第1张图片

 

 

这种直接返回错误页面,对于用户而言,显然是太不友好了哈!

2.2 统一异常处理

java异常详解

首先,定义自己的异常类,随便起个名字哈,MyException.java

@Data

public class MyException extends Exception{
 
 private Integer code;
 private String Message;
 
 public MyException(Integer code,String Message) {
 this.code = code;
 this.Message = Message;
 }
}

然后定义自己的异常处理类,ExceptionHandle.java

如果想学习Java工程化、高性能及分布式、深入浅出。微服务、Spring,MyBatis,Netty源码分析的朋友可以加我的Java高级交流:787707172,群里有阿里大牛直播讲解技术,以及Java大型互联网技术的视频免费分享给大家。

如果返回的对象是JSON的话,可以用@RestControllerAdvice

@ControllerAdvice

public class ExceptionHandle {
 
 private final static Logger logger = LoggerFactory.getLogger(ExceptionHandle.class);
 
 @ExceptionHandler(value = Exception.class)
 @ResponseBody
 public Result handle(Exception e) {
 if (e instanceof MyException) {
 MyException myException = (MyException) e;
 return ResultUtil.error(boyException.getCode(), boyException.getMessage());
 }else {
 logger.error("【系统异常】{}", e);
 return new Result(-1, "未知错误");
 }
 }
}

3、统一异常处理源码解析

3.1 注解源码解析

java注解详解

@ControllerAdvice

@ExceptionHandler

@RestControllerAdvice与@ExceptionHandler注解是sprngmvc中与异常捕获与处理相关的注解,它的入口也是DispatcherServlet中的doDispatcher()方法中,如下:

this.processDispatchResult(processedRequest, 
response, mappedHandler, mv, (Exception)dispatchException);

后面会进入HandlerExceptionResolverComposite的resolveException方法,这个ExceptionHandlerResolverComposite包含三个ExcpetionHandlerResolver,是在springmvc中生成的,在springboot中其生成代码如下:

 

@Bean
	public HandlerExceptionResolver handlerExceptionResolver() {
		List exceptionResolvers = new ArrayList<>();
		configureHandlerExceptionResolvers(exceptionResolvers);
		if (exceptionResolvers.isEmpty()) {
			addDefaultHandlerExceptionResolvers(exceptionResolvers);
		}
		extendHandlerExceptionResolvers(exceptionResolvers);
		HandlerExceptionResolverComposite composite = new HandlerExceptionResolverComposite();
		composite.setOrder(0);
		composite.setExceptionResolvers(exceptionResolvers);
		return composite;
	}

后面他会进入ExceptionHandlerExceptionResolver类的方法:

protected ModelAndView doResolveHandlerMethodException(HttpServletRequest request,
			HttpServletResponse response, @Nullable HandlerMethod handlerMethod,
 Exception exception) {
 
		 ServletInvocableHandlerMethod exceptionHandlerMethod = 
 getExceptionHandlerMethod(handlerMethod, exception);
 }

在这个方法中的第一行,getExceptionHandlerMethod方法,其进行了查找对应的带有@ControllerAdvice注解的类型和对应匹配的方法,然后在doResolverHandlerMethod方法中进行了处理,这就是整个流程。

@ControllerAdvice的加载过程:

首先在springboot扫描的时候,会把@ControllerAdvice的bean放入到beanFactory里面去,此时只要从beanFactory中获取到需要的bean即可,处理方式在ExceptionHandlerExceptionResolver类中:

@Override
	public void afterPropertiesSet() {
		// Do this first, it may add ResponseBodyAdvice beans
		initExceptionHandlerAdviceCache();
	private void initExceptionHandlerAdviceCache() {
		if (getApplicationContext() == null) {
			return;
		}
		if (logger.isDebugEnabled()) {
			logger.debug("Looking for exception mappings: " + getApplicationContext());
		}
 
		List adviceBeans = ControllerAdviceBean.findAnnotatedBeans(getApplicationContext());

封装好后,获取带有@Exceptionhandler的注解方法,即根据异常类型进行调用了

你可能感兴趣的:(Java数据校验详解&springboot 统一异常处理(包含统一数据校验))