springboot 全局转换器和参数校验

前言


我们在springboot 项目中只要实现convert接口就可以对前台传过来的参数就行所需要的转化,全局转换只适用于get请求
比如string转date,如下

@Component
public class StringToDateConverter implements Converter<String, Date> {

    @Override
    public Date convert(String source) {
        if (StringUtils.isBlank(source)) {
            return null;
        }
        if (source.matches("^\\d{4}-\\d{1,2}-\\d{1,2}$")) {
            return parseDate(source.trim(), "yyyy-MM-dd");
        }
        if (source.matches("^\\d{4}-\\d{1,2}-\\d{1,2} {1}\\d{1,2}:\\d{1,2}:\\d{1,2}$")) {
            return parseDate(source.trim(), "yyyy-MM-dd HH:mm:ss");
        }
        throw new IllegalArgumentException("Invalid value '" + source + "'");
    }

    public Date parseDate(String dateStr, String format) {
        Date date = null;
        try {
            date = new SimpleDateFormat(format).parse(dateStr);
        }
        catch (ParseException e) {
            log.warn("转换{}为日期(pattern={})错误!", dateStr, format);
        }
        return date;
    }
    
}

其次我们在接口入参中 加@Validated 注解就可以实现对接口dto参数的校验

 public static class BasePageDto {
        /**
         * 基准货币
         */
        @NotNull(message = "基础id不能为空!")
        private Integer baseId;

本文从源码角度分析下这些工作spring(或者说springboot)是如何帮我们完成的。

原理


转化器的注册


先看WebMvcAutoConfiguration ,这个是springboot 注册SpringMVC相关处理器的自动配置类,WebMvcAutoConfiguration中有3个内部类WebMvcAutoConfigurationAdapter,EnableWebMvcConfiguration,ResourceChainCustomizerConfiguration
然后看

        @Configuration
    @Import(EnableWebMvcConfiguration.class)
    @EnableConfigurationProperties({ WebMvcProperties.class, ResourceProperties.class })
    public static class WebMvcAutoConfigurationAdapter extends WebMvcConfigurerAdapter {

对比我们在spring mvc 中常常写的WebConfig (如下)是不是非常的一致,spring 的自动配置实际上跟springmvc我们手动配置是一致的。

@Configuration
@EnableWebMvc
@ComponentScan(basePackages= "com.qijun.spring.demo.controller")
public class WebConfig extends WebMvcConfigurerAdapter{
}

@Import(EnableWebMvcConfiguration.class) 中的EnableWebMvcConfiguration.class 是WebMvcAutoConfiguration 中的内部静态类,作用与@EnableWebMvc 相同,WebMvcAutoConfigurationAdapter 中是springboot 默认配置 是对WebMvcConfigurer的重写,然后可以看到addFormatters ,这个是注册自定义convert的方法的入口

    @Override
        public void addFormatters(FormatterRegistry registry) {
                  // 添加自定义的converter
            for (Converter<?, ?> converter : getBeansOfType(Converter.class)) {
                registry.addConverter(converter);
            }
            for (GenericConverter converter : getBeansOfType(GenericConverter.class)) {
                registry.addConverter(converter);
            }
            for (Formatter<?> formatter : getBeansOfType(Formatter.class)) {
                registry.addFormatter(formatter);
            }
        }

然后在看 WebMvcConfigurationSupport 这个类,先看这个类的继承关系,它是EnableWebMvcConfiguration 这个类的父类


image.png

WebMvcConfigurationSupport 中看到如下代码,也就是在生成DefaultFormattingConversionService 这个类型的conversionService 这个bean时会把对应的 converter 注册 到这个conversionService

@Bean
    public FormattingConversionService mvcConversionService() {
        FormattingConversionService conversionService = new DefaultFormattingConversionService();
        //实际调用的是DelegatingWebMvcConfiguration 中configurers 的addFormatters方法
              addFormatters(conversionService);
        return conversionService;
    }

image.png
public class FormattingConversionService extends GenericConversionService
        implements FormatterRegistry, EmbeddedValueResolverAware

DefaultFormattingConversionService 从FormattingConversionService ,FormattingConversionService 又实现了FormatterRegistry接口,addFormatters 的入参也是FormatterRegistry 类型的。

然后再回到WebMvcConfigurationSupport 那个继承关系图,中间的DelegatingWebMvcConfiguration,这个类中有

private final WebMvcConfigurerComposite configurers = new WebMvcConfigurerComposite();
@Autowired(required = false)
    public void setConfigurers(List<WebMvcConfigurer> configurers) {
        if (!CollectionUtils.isEmpty(configurers)) {
            this.configurers.addWebMvcConfigurers(configurers);
        }
    }

configurers 就是所有webmvc的配置类的集合,注释写的很清楚就是1个或者多个WebMvcConfigurer

/**
 * A {@link WebMvcConfigurer} that delegates to one or more others.
 *
 * @author Rossen Stoyanchev
 * @since 3.1
 */
class WebMvcConfigurerComposite implements WebMvcConfigurer {

再看DelegatingWebMvcConfiguration的setConfigurers方法,会把当前所有的WebMvcConfigurer都加到configurers 里去,也包括springboot 提供的WebMvcAutoConfigurationAdapter 这个配置类,就串上了,这样就完成了把我们自定义的converter加到了conversionService 中。最后GenericConversionService调用addConverter方法就不分析了。

转化器的调用


spring mvc 请求处理的流程 如下


image.png

最关键的关于请求参数处理的代码在ServletInvocableHandlerMethod 这个类中
首先是invokeForRequest

public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer,
            Object... providedArgs) throws Exception {
        // 使用反射调用接口方法
        Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs);
        setResponseStatus(webRequest);
public Object invokeForRequest(NativeWebRequest request, ModelAndViewContainer mavContainer,
            Object... providedArgs) throws Exception {
                // 获取接口参数的参数值
        Object[] args = getMethodArgumentValues(request, mavContainer, providedArgs);
        if (logger.isTraceEnabled()) {
            logger.trace("Invoking '" + ClassUtils.getQualifiedMethodName(getMethod(), getBeanType()) +
                    "' with arguments " + Arrays.toString(args));
        }
                //真正使用反射调用接口方法
        Object returnValue = doInvoke(args);
        if (logger.isTraceEnabled()) {
            logger.trace("Method [" + ClassUtils.getQualifiedMethodName(getMethod(), getBeanType()) +
                    "] returned [" + returnValue + "]");
        }
        return returnValue;
    }

主要看getMethodArgumentValues 这个方法
spring处理请求的时候,会根据ServletInvocableHandlerMethod的属性argumentResolvers
(这个属性 是它的父类InvocableHandlerMethod中定义的)进行处理,其中argumentResolvers属性是一个
HandlerMethodArgumentResolverComposite类(这里使用了组合模式的一种变形),这个类是实现了HandlerMethodArgumentResolver接口的类,
里面有各种实现了HandlerMethodArgumentResolver的List集合。
常见的HandlerMethodArgumentResolver 的实现类

  1. RequestParamMethodArgumentResolver
    支持带有@RequestParam注解的参数或带有MultipartFile类型的参数

  2. RequestParamMapMethodArgumentResolver
    支持带有@RequestParam注解的参数 && @RequestParam注解的属性value存在 && 参数类型是实现Map接口的属性

  3. PathVariableMethodArgumentResolver
    支持带有@PathVariable注解的参数 且如果参数实现了Map接口,@PathVariable注解需带有value属性

  4. MatrixVariableMethodArgumentResolver
    支持带有@MatrixVariable注解的参数 且如果参数实现了Map接口,@MatrixVariable注解需带有value属性

  5. ServletModelAttributeMethodProcessor
    默认的argumentResolvers实例化的时候 两个ServletModelAttributeMethodProcessor,属性annotationNotRequired一个为true,1个为false。为true的ServletModelAttributeMethodProcessor处理带@ModelAttribute注解的参数,annotationNotRequired属性为false,处理非简单类型参数,最终通过DataBinder实例化类型对象,并写入对应的属性。

  6. ErrorsMethodArgumentResolver
    后面我们会看到,处理BindingResult 类型入参

  7. RequestResponseBodyMethodProcessor
    处理requestBody类型的请求

private Object[] getMethodArgumentValues(NativeWebRequest request, ModelAndViewContainer mavContainer,
            Object... providedArgs) throws Exception {

        MethodParameter[] parameters = getMethodParameters();
        Object[] args = new Object[parameters.length];
        for (int i = 0; i < parameters.length; i++) {
            MethodParameter parameter = parameters[i];
            parameter.initParameterNameDiscovery(this.parameterNameDiscoverer);
            args[i] = resolveProvidedArgument(parameter, providedArgs);
            if (args[i] != null) {
                continue;
            }
            if (this.argumentResolvers.supportsParameter(parameter)) {
                try {
                      //此处循环解析参数,断点1
                    // 根据参数类型调用特定的HandlerMethodArgumentResolver实现类处理参数
                    args[i] = this.argumentResolvers.resolveArgument(
                            parameter, mavContainer, request, this.dataBinderFactory);
                    continue;
                }
                catch (Exception ex) {
                    if (logger.isDebugEnabled()) {
                        logger.debug(getArgumentResolutionErrorMessage("Failed to resolve", i), ex);
                    }
                    throw ex;
                }
            }
            if (args[i] == null) {
                throw new IllegalStateException("Could not resolve method parameter at index " +
                        parameter.getParameterIndex() + " in " + parameter.getMethod().toGenericString() +
                        ": " + getArgumentResolutionErrorMessage("No suitable resolver for", i));
            }
        }
        return args;
    }

然后是argumentResolvers 处理过程,从下图红框开始。ModelAttributeMethodProcessor 实现了HandlerMethodArgumentResolver。


5641667-2e04deb7990ef00d.png

下面通过一个简单的接口来分析下

 @ApiOperation(value = "testRequestBody")
    @RequestMapping(value = "/testRequestBody",method = RequestMethod.GET)
    public void testMap(@Validated InputBody input, BindingResult BindingResult) {
        System.out.println(input.getDate() + "   " + input.getDate());
    }
@Data
public class InputBody {
    @NotNull
    private Date date;
}

我们分别在getMethodArgumentValues 的this.argumentResolvers.resolveArgument 打断点
和HandlerMethodArgumentResolverComposite 的resolveArgument 方法处打断点
还有自定义的convert 的convert打断点


image.png

spring在处理第一个参数


image.png

可以非常明确的看到第一参数对应的是ServletModelAttributeMethodProcessor 参数处理类
image.png

最后会在ModelAttributeMethodProcessor 的bindRequestParameters 通过一系列的步骤如上图,找到我们之前注册的convert,然后转换。

下面简单分析下ModelAttributeMethodProcessor resolveArgument 方法,ServletModelAttributeMethodProcessor 是ModelAttributeMethodProcessor 的子类 前面提到的annotationNotRequired 是在ModelAttributeMethodProcessor 里的

@Override
    public final Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer,
            NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {

        String name = ModelFactory.getNameForParameter(parameter);
        ModelAttribute ann = parameter.getParameterAnnotation(ModelAttribute.class);
        if (ann != null) {
            mavContainer.setBinding(name, ann.binding());
        }
                // 创建空的参数属性对象实例
        Object attribute = (mavContainer.containsAttribute(name) ? mavContainer.getModel().get(name) :
                createAttribute(name, parameter, binderFactory, webRequest));
                 //获取webdateBinder对象
        WebDataBinder binder = binderFactory.createBinder(webRequest, attribute, name);
        if (binder.getTarget() != null) {
            if (!mavContainer.isBindingDisabled(name)) {
                                //绑定参数
                bindRequestParameters(binder, webRequest);
            }
                        // 如果需要使用validate校验(使用了@Validated注解),获取校验结果
            validateIfApplicable(binder, parameter);
                        // 判断参数校验是否有错误,是否有bindingResult参数
            if (binder.getBindingResult().hasErrors() && isBindExceptionRequired(binder, parameter)) {
                throw new BindException(binder.getBindingResult());
            }
        }

        // 最后把处理好的属性和bindingResult 放入ModelAndView对象
        Map<String, Object> bindingResultModel = binder.getBindingResult().getModel();
        mavContainer.removeAttributes(bindingResultModel);
        mavContainer.addAllAttributes(bindingResultModel);

        return binder.convertIfNecessary(binder.getTarget(), parameter.getParameterType(), parameter);
    }

最后回到前面的处理参数的循环中处理第二个参数,这个参数使用的是ErrorsMethodArgumentResolver处理类,参数的值是在之前获取的ModelAndView对象取的最后一个元素。


image.png

一个注意点
如果BindingResult bindingResult不在请求参数的后一个,是不能获取这个校验结果的,源码如下,在 validateIfApplicable(binder, parameter)之后

// AbstractMessageConverterMethodArgumentResolver
    protected boolean isBindExceptionRequired(WebDataBinder binder, MethodParameter methodParam) {
        int i = methodParam.getParameterIndex();
        Class<?>[] paramTypes = methodParam.getMethod().getParameterTypes();
        boolean hasBindingResult = (paramTypes.length > (i + 1) && Errors.class.isAssignableFrom(paramTypes[i + 1]));
        return !hasBindingResult;
    }

参考


https://www.cnblogs.com/sunny3096/p/7215906.html
http://blog.csdn.net/u012410733/article/details/53368351
http://blog.csdn.net/u012410733/article/details/51920055

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,236评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,867评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,715评论 0 340
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,899评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,895评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,733评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,085评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,722评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,025评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,696评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,816评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,447评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,057评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,254评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,204评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,561评论 2 343

推荐阅读更多精彩内容