研究javax.validation.constraints.NotNull运行原理

2020-04-24 17:53:44 浏览数 (1)

1. 研究javax.validation.constraints.NotNull运行原理

1.1. 源码跟进

  1. 为了找到NotNull到底是在哪里被处理,我先打印一个错误信息,根据错误信息的关键字,我首先找到的是这个地方,可以看到是webmvc包,可以理解,毕竟是通过接口请求并拦截的,需要经过webmvc
  1. 接下来要探究这句话是怎么产生的了,追溯body值得来源,发现最终捕获异常的地方在DispatcherServlet类的如下方法,感觉快要找到了,进入handle方法吧
  1. 继续往里面找,找到了抛MethodArgumentNotValidException异常的地方,可以看到验证参数的方法就是validateIfApplicable
  1. 进到这个方法就能看到熟悉的小伙伴了,@Validated这个注解,在需要验证的Controller接口都需要加,之后的核心验证方法为binder.validate,之后需要层层递进分叉非常多了,我就讲一条我遇到的实际问题找寻源码的路径

1.2. 问题

  1. 只是在controller层,写个@Validated注解,之后的@NotNull判断等注解判断只在@Validated定义的对象生效,现在我想要实现对象中的对象也实现验证效果,我这里直接说结论了,我写了如下类,我需要body对象也能够得到参数验证,则在类方法上加个@Valid注解即可实现,接下去看源码
代码语言:javascript复制
@Data
public class Request<T> {
    /**
     * 请求体
     */
    @Valid
    private T body;

    /**
     * 请求码(预留可不填)
     */
    private Integer requestCode;

    /**
     * 额外请求参数,可另做处理
     */
    private Map<String,Object> extend;
}
  1. 由于深入层数过多,我会挑几个重点截图,起到抛砖引玉的作用了,想要深入了解一定要自己看源码
  1. 通过SpringValidatorAdapter验证类作为核心
  1. 接下去解析注解,中间调整过多,我会跳过几层
  1. 终于找到获取对象属性的@Valid注解
  1. 获得了这个级联元数据,后续的判断就会用到这个,设值valueContext,我把这个ExampleDeleteVo对象的id属性设值了注解@NotNull
  1. 可以看到它获取了id为null的值放入了currentValue

1.3. 总结

一开始我研究@Validated注解就是为了找是否有办法验证对象内对象,如果不行可能就需要自己写拦截器方法了,不到迫不得已我也不想重复造轮子,毕竟@Validated自带的验证这么多,写起来也蛮累的,还容易出bug。有耐心看完这篇文章的估计是遇到@Vaildated的问题了,希望能起到抛砖引玉的作用吧

0 人点赞