SpringBoot的HttpMessageConverter使用(2)@RequestBody和@ResponseBody中的应用

  1. HttpMessageConverter加载流程
    1.1 mvc默认读取所有WebMvcConfigurer配置类的消息转换器
    1.2 mvc默认的WebMvcConfigurer配置类
    1.3 默认HttpMessageConverters加载消息转换器
    1.4 WebMvcConfigurationSupport获取消息转换器
    1.5 SpringMVC消息转换器总结
  2. 扩展自定义消息转换器
    2.1 方式一:修改默认的消息转换器(不推荐使用@EnableWebMvc注解)
    2.2 方式二:将自定义的消息转换器放入到Spring容器中

SpringMVC中,只需要在Controller方法上加入@RequestBody@ResponseBody注解便可以完成请求参数和响应参数的序列化和反序列化。那么消息转换器是如何工作的?

1. HttpMessageConverter加载流程

识别流程.png

消息转化器真正被使用到的源码位置:org.springframework.web.servlet.mvc.method.annotation.AbstractMessageConverterMethodProcessor#writeWithMessageConverters(T, org.springframework.core.MethodParameter, org.springframework.http.server.ServletServerHttpRequest, org.springframework.http.server.ServletServerHttpResponse)

1.1 mvc默认读取所有WebMvcConfigurer配置类的消息转换器

关键bean:DelegatingWebMvcConfiguration

  1. 启动项目后,加载DelegatingWebMvcConfiguration配置,读取容器中所有的WebMvcConfigurer的bean。
@Configuration(proxyBeanMethods = false)
public class DelegatingWebMvcConfiguration extends WebMvcConfigurationSupport {

    private final WebMvcConfigurerComposite configurers = new WebMvcConfigurerComposite();


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

初始化EnableWebMvcConfiguration类时,会创建RequestMappingHandlerAdapter的bean。

    @Configuration(proxyBeanMethods = false)
    public static class EnableWebMvcConfiguration extends DelegatingWebMvcConfiguration implements ResourceLoaderAware {
        @Bean
        @Override
        public RequestMappingHandlerAdapter requestMappingHandlerAdapter(
                @Qualifier("mvcContentNegotiationManager") ContentNegotiationManager contentNegotiationManager,
                @Qualifier("mvcConversionService") FormattingConversionService conversionService,
                @Qualifier("mvcValidator") Validator validator) {
            RequestMappingHandlerAdapter adapter = super.requestMappingHandlerAdapter(contentNegotiationManager,
                    conversionService, validator);
            adapter.setIgnoreDefaultModelOnRedirect(
                    this.mvcProperties == null || this.mvcProperties.isIgnoreDefaultModelOnRedirect());
            return adapter;
        }

    }

在创建RequestMappingHandlerAdapter时,会调用DelegatingWebMvcConfiguration类的getMessageConverters()方法获取到消息转换器。

而DelegatingWebMvcConfiguration类实现了configureMessageConverters()方法,会读取所以的WebMvcConfigurer配置类中的消息转换器的类型。

org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport#getMessageConverters父类的实现逻辑。

    protected final List<HttpMessageConverter<?>> getMessageConverters() {
        if (this.messageConverters == null) {
            this.messageConverters = new ArrayList<>();
          //DelegatingWebMvcConfiguration实现了该逻辑。
            configureMessageConverters(this.messageConverters);
            if (this.messageConverters.isEmpty()) {
                addDefaultHttpMessageConverters(this.messageConverters);
            }
            extendMessageConverters(this.messageConverters);
        }
        return this.messageConverters;
    }

最终调用的是:org.springframework.web.servlet.config.annotation.WebMvcConfigurerComposite#configureMessageConverters方法。

    @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        for (WebMvcConfigurer delegate : this.delegates) {
            delegate.configureMessageConverters(converters);
        }
    }

1.2 mvc默认的WebMvcConfigurer配置类

    public static class WebMvcAutoConfigurationAdapter implements WebMvcConfigurer {
       private final ObjectProvider<HttpMessageConverters> messageConvertersProvider;
        //默认的WebMvcConfigurer完成消息转换器的配置。会去初始化HttpMessageConverters对象,并获取HttpMessageConvertersbean中的消息转换器
        @Override
        public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
            this.messageConvertersProvider
                    .ifAvailable((customConverters) -> converters.addAll(customConverters.getConverters()));
        }
    }

1.3 默认HttpMessageConverters加载消息转换器

public class HttpMessageConvertersAutoConfiguration {
   //项目启动后,加载Spring容器中所有的消息转换器
    @Bean
    @ConditionalOnMissingBean
    public HttpMessageConverters messageConverters(ObjectProvider<HttpMessageConverter<?>> converters) {
        return new HttpMessageConverters(converters.orderedStream().collect(Collectors.toList()));
    }
}
image.png

执行构造方法,读取消息转换器到属性中。

public class HttpMessageConverters implements Iterable<HttpMessageConverter<?>> {
    private final List<HttpMessageConverter<?>> converters;
    //构造方法,此时传入的消息转换器为Spring容器中的
    public HttpMessageConverters(Collection<HttpMessageConverter<?>> additionalConverters) {
        this(true, additionalConverters);
    }
    //参数1为true:表示会加载默认的消息转换器
    public HttpMessageConverters(boolean addDefaultConverters, Collection<HttpMessageConverter<?>> converters) {
         //核心方法getCombinedConverters,将自定义消息转换器和默认的消息转换器排序。
        //由于参数为true,那么会调用getDefaultConverters()获取默认的消息转换器。
        List<HttpMessageConverter<?>> combined = getCombinedConverters(converters,
                addDefaultConverters ? getDefaultConverters() : Collections.emptyList());
        combined = postProcessConverters(combined);
        this.converters = Collections.unmodifiableList(combined);
    }
    //如果自定义转换器可以替换默认转换器,那么自定义转换器放在前面。
    private List<HttpMessageConverter<?>> getCombinedConverters(Collection<HttpMessageConverter<?>> converters,
            List<HttpMessageConverter<?>> defaultConverters) {
        List<HttpMessageConverter<?>> combined = new ArrayList<>();
        List<HttpMessageConverter<?>> processing = new ArrayList<>(converters);
        for (HttpMessageConverter<?> defaultConverter : defaultConverters) {
            Iterator<HttpMessageConverter<?>> iterator = processing.iterator();
            while (iterator.hasNext()) {
                HttpMessageConverter<?> candidate = iterator.next();
                if (isReplacement(defaultConverter, candidate)) {
                    combined.add(candidate);
                    iterator.remove();
                }
            }
            combined.add(defaultConverter);
            if (defaultConverter instanceof AllEncompassingFormHttpMessageConverter) {
                configurePartConverters((AllEncompassingFormHttpMessageConverter) defaultConverter, converters);
            }
        }
        combined.addAll(0, processing);
        return combined;
    }
    //判断自定义的消息转换器是否可以替换默认的消息转换器
    private boolean isReplacement(HttpMessageConverter<?> defaultConverter, HttpMessageConverter<?> candidate) {
        for (Class<?> nonReplacingConverter : NON_REPLACING_CONVERTERS) {
            if (nonReplacingConverter.isInstance(candidate)) {
                return false;
            }
        }
        return ClassUtils.isAssignableValue(defaultConverter.getClass(), candidate);
    }
    //获取默认的消息转换器
    private List<HttpMessageConverter<?>> getDefaultConverters() {
        List<HttpMessageConverter<?>> converters = new ArrayList<>();
        //判断是否引入了WebMvcConfigurationSupport类的包,mvc一般会加载
        if (ClassUtils.isPresent("org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport",
                null)) {
            converters.addAll(new WebMvcConfigurationSupport() {

                public List<HttpMessageConverter<?>> defaultMessageConverters() {
                    return super.getMessageConverters();
                }

            }.defaultMessageConverters());
        }
        else {
            //使用RestTemplate中定义的消息转换器。
            converters.addAll(new RestTemplate().getMessageConverters());
        }
        //重排序,将xml消息转换器放在最后
        reorderXmlConvertersToEnd(converters);
        return converters;
    }
    //将xml的消息转换器放在最后
    private void reorderXmlConvertersToEnd(List<HttpMessageConverter<?>> converters) {
        List<HttpMessageConverter<?>> xml = new ArrayList<>();
        for (Iterator<HttpMessageConverter<?>> iterator = converters.iterator(); iterator.hasNext();) {
            HttpMessageConverter<?> converter = iterator.next();
            if ((converter instanceof AbstractXmlHttpMessageConverter)
                    || (converter instanceof MappingJackson2XmlHttpMessageConverter)) {
                xml.add(converter);
                iterator.remove();
            }
        }
        converters.addAll(xml);
    }
}
  1. 读取默认的消息转换器(默认优先读取WebMvcConfigurationSupport类自定义的)。
  2. 将默认的消息转换器重排序(xml消息转换器放在后面)。
  3. 自定义的消息转换器和默认的消息转换器排序(比较自定义转换器类型是否为可以替换默认转换器的类型,如果是,将自定义转换器放在默认转换器的前面)。

1.4 WebMvcConfigurationSupport获取消息转换器

WebMvcConfigurationSupport提供了getMessageConverters()方法。

  1. 若子类重写了configureMessageConverters方法并返回消息转换器,那么不会只想addDefaultHttpMessageConverters去获取默认的消息转换器;
  2. 子类可重写extendMessageConverters方法,扩展消息转换器;
public class WebMvcConfigurationSupport implements ApplicationContextAware, ServletContextAware {
    
   //依旧是判断依赖是否引入,判断是否去加载消息转换器。
    static {
        ClassLoader classLoader = WebMvcConfigurationSupport.class.getClassLoader();
        romePresent = ClassUtils.isPresent("com.rometools.rome.feed.WireFeed", classLoader);
        jaxb2Present = ClassUtils.isPresent("javax.xml.bind.Binder", classLoader);
        jackson2Present = ClassUtils.isPresent("com.fasterxml.jackson.databind.ObjectMapper", classLoader) &&
                ClassUtils.isPresent("com.fasterxml.jackson.core.JsonGenerator", classLoader);
        jackson2XmlPresent = ClassUtils.isPresent("com.fasterxml.jackson.dataformat.xml.XmlMapper", classLoader);
        jackson2SmilePresent = ClassUtils.isPresent("com.fasterxml.jackson.dataformat.smile.SmileFactory", classLoader);
        jackson2CborPresent = ClassUtils.isPresent("com.fasterxml.jackson.dataformat.cbor.CBORFactory", classLoader);
        gsonPresent = ClassUtils.isPresent("com.google.gson.Gson", classLoader);
        jsonbPresent = ClassUtils.isPresent("javax.json.bind.Jsonb", classLoader);
    }
      
    //获取默认的消息转换器
    protected final List<HttpMessageConverter<?>> getMessageConverters() {
        if (this.messageConverters == null) {
            this.messageConverters = new ArrayList<>();
            //子类可以重写该方法,定义消息转换器。
            configureMessageConverters(this.messageConverters);
            if (this.messageConverters.isEmpty()) {
                //若未定义的消息转换器,那么会走该方法去增加消息转换器。
                addDefaultHttpMessageConverters(this.messageConverters);
            }
            //子类可以重写该方法,扩展消息转换器
            extendMessageConverters(this.messageConverters);
        }
        return this.messageConverters;
    }
    //根据依赖是否存在,去增加默认的消息转换器。
    protected final void addDefaultHttpMessageConverters(List<HttpMessageConverter<?>> messageConverters) {
        messageConverters.add(new ByteArrayHttpMessageConverter());
        messageConverters.add(new StringHttpMessageConverter());
        messageConverters.add(new ResourceHttpMessageConverter());
        messageConverters.add(new ResourceRegionHttpMessageConverter());
        try {
            messageConverters.add(new SourceHttpMessageConverter<>());
        }
        catch (Throwable ex) {
            // Ignore when no TransformerFactory implementation is available...
        }
        messageConverters.add(new AllEncompassingFormHttpMessageConverter());

        if (romePresent) {
            messageConverters.add(new AtomFeedHttpMessageConverter());
            messageConverters.add(new RssChannelHttpMessageConverter());
        }

        if (jackson2XmlPresent) {
            Jackson2ObjectMapperBuilder builder = Jackson2ObjectMapperBuilder.xml();
            if (this.applicationContext != null) {
                builder.applicationContext(this.applicationContext);
            }
            messageConverters.add(new MappingJackson2XmlHttpMessageConverter(builder.build()));
        }
        else if (jaxb2Present) {
            messageConverters.add(new Jaxb2RootElementHttpMessageConverter());
        }

        if (jackson2Present) {
            Jackson2ObjectMapperBuilder builder = Jackson2ObjectMapperBuilder.json();
            if (this.applicationContext != null) {
                builder.applicationContext(this.applicationContext);
            }
            messageConverters.add(new MappingJackson2HttpMessageConverter(builder.build()));
        }
        else if (gsonPresent) {
            messageConverters.add(new GsonHttpMessageConverter());
        }
        else if (jsonbPresent) {
            messageConverters.add(new JsonbHttpMessageConverter());
        }

        if (jackson2SmilePresent) {
            Jackson2ObjectMapperBuilder builder = Jackson2ObjectMapperBuilder.smile();
            if (this.applicationContext != null) {
                builder.applicationContext(this.applicationContext);
            }
            messageConverters.add(new MappingJackson2SmileHttpMessageConverter(builder.build()));
        }
        if (jackson2CborPresent) {
            Jackson2ObjectMapperBuilder builder = Jackson2ObjectMapperBuilder.cbor();
            if (this.applicationContext != null) {
                builder.applicationContext(this.applicationContext);
            }
            messageConverters.add(new MappingJackson2CborHttpMessageConverter(builder.build()));
        }
    }
}

1.5 SpringMVC消息转换器总结(重要)

SpringBoot默认会加载Spring容器中定义的消息转换器(由开发者定义)+默认配置的消息转换器(比如xml转换器便是根据pom依赖生成的)。并进行排序(例如json消息转换器优先级高于xml消息转换器;比较自定义转换器类型是否为可以替换默认转换器的类型,如果是,将自定义转换器放在默认转换器的前面)。

2. 扩展自定义消息转换器

2.1 方式一:修改默认的消息转换器(不推荐使用@EnableWebMvc注解)

@Configuration
//@EnableWebMvc  //有坑
public class MessageConverterConfig implements WebMvcConfigurer {
    /**
     * 配置消息转换器--这里我用的是alibaba 开源的 fastjson
     *
     * @param converters
     */
    @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {

        //1.需要定义一个convert转换消息的对象;
        FastJsonHttpMessageConverter fastJsonHttpMessageConverter = new FastJsonHttpMessageConverter();
        //2.添加fastJson的配置信息,比如:是否要格式化返回的json数据;
        FastJsonConfig fastJsonConfig = new FastJsonConfig();
        fastJsonConfig.setSerializerFeatures(
                SerializerFeature.WriteMapNullValue,
                SerializerFeature.WriteNullNumberAsZero,
                SerializerFeature.QuoteFieldNames,
                SerializerFeature.WriteNullStringAsEmpty,
                SerializerFeature.DisableCircularReferenceDetect,
                SerializerFeature.WriteNullListAsEmpty,
                SerializerFeature.WriteDateUseDateFormat);
        //3处理中文乱码问题
        List<MediaType> fastMediaTypes = new ArrayList<>();
        fastMediaTypes.add(MediaType.APPLICATION_JSON);
        //4.在convert中添加配置信息.
        fastJsonHttpMessageConverter.setSupportedMediaTypes(fastMediaTypes);
        fastJsonHttpMessageConverter.setFastJsonConfig(fastJsonConfig);
        //5.将convert添加到converters当中.
        converters.add(0, fastJsonHttpMessageConverter);
    }
}
image.png

在截图中可以发现:定义的消息转换器排在了首位。

不推荐使用@EnableWebMvc注解,若是使用该注解后,mvc提供默认的WebMvcConfigurer配置类WebMvcAutoConfigurationAdapter不会加载到Spring容器中,也就不会将HttpMessageConverters的消息转换器加载到程序中。

2.2 方式二:将自定义的消息转换器放入到Spring容器中。

本质依旧借助默认的WebMvcAutoConfigurationAdapter配置,不能在项目中声明@EnableWebMvc注解,否则该方式不生效,本质借助了HttpMessageConverters类的getCombinedConverters()方法将自定义的消息转换器排在了默认的消息转换器之上。

@Configuration
public class MessageConverterConfig2 {

    @Bean
    public FastJsonHttpMessageConverter fastJsonHttpMessageConverter() {
        //1.需要定义一个convert转换消息的对象;
        FastJsonHttpMessageConverter fastJsonHttpMessageConverter = new FastJsonHttpMessageConverter();
        //2.添加fastJson的配置信息,比如:是否要格式化返回的json数据;
        FastJsonConfig fastJsonConfig = new FastJsonConfig();
        fastJsonConfig.setSerializerFeatures(
                SerializerFeature.WriteMapNullValue,
                SerializerFeature.WriteNullNumberAsZero,
                SerializerFeature.QuoteFieldNames,
                SerializerFeature.WriteNullStringAsEmpty,
                SerializerFeature.DisableCircularReferenceDetect,
                SerializerFeature.WriteNullListAsEmpty,
                SerializerFeature.WriteDateUseDateFormat);
        //3处理中文乱码问题
        List<MediaType> fastMediaTypes = new ArrayList<>();
        fastMediaTypes.add(MediaType.APPLICATION_JSON);
        //4.在convert中添加配置信息.
        fastJsonHttpMessageConverter.setSupportedMediaTypes(fastMediaTypes);
        fastJsonHttpMessageConverter.setFastJsonConfig(fastJsonConfig);
        return fastJsonHttpMessageConverter;
    }
}
@Service
public class NotNullMappingJackson2HttpMessageConverter extends MappingJackson2HttpMessageConverter {

    @Override
    public ObjectMapper getObjectMapper() {
        return super.getObjectMapper().setSerializationInclusion(JsonInclude.Include.NON_NULL);
    }
}
image.png
  1. 自定义的消息转换器优先级比可以替换的默认的消息转换器,即NotNullMappingJackson2HttpMessageConverter优先级排在MappingJackson2HttpMessageConverter优先级之上。

  2. 若自定义的消息转换器不能替换默认的消息转换器,那么将会放在集合的最上面【优先级最高】,例如FastJsonHttpMessageConverter。

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

推荐阅读更多精彩内容