spring mvc HttpMessageConverter 消息转换器

发布于:2025-08-15 ⋅ 阅读:(15) ⋅ 点赞:(0)

HttpMessageConverter 主要在 Spring MVC 中用于处理 HTTP 請求和响应。它的核心作用是将 HTTP请求体转换成 Java对象,以及將 Java对象转换成 HTTP响应体。这两个场景都是在 @RequestBody / @ResponseBody /注解修饰的请求方法,或HttpEntity类型接口参数上 触发的。

读取请求体(Request → Java 对象)

  • 触发条件:

    • Controller 方法参数上有 @RequestBody
    • 或参数类型是 HttpEntity<T> / RequestEntity<T>
  • 处理流程:

    1. 进入 RequestResponseBodyMethodProcessor#resolveArgument
    2. 调用 readWithMessageConverters(...)(父类 AbstractMessageConverterMethodArgumentResolver
    3. 遍历所有 HttpMessageConverter,找第一个 canRead() 返回 true
    4. 调用该 Converter 的 read() 方法反序列化请求体到 Java 对象

写入响应体(Java 对象 → Response)

  • 触发条件:

    • Controller 方法有 @ResponseBody
    • 或类上有 @RestController
    • 或返回类型是 HttpEntity<T> / ResponseEntity<T>
  • 处理流程:

    1. 进入 RequestResponseBodyMethodProcessor#handleReturnValue
    2. 调用 writeWithMessageConverters(...)(父类 AbstractMessageConverterMethodProcessor
    3. 遍历所有 HttpMessageConverter,找第一个 canWrite() 返回 true
    4. 调用该 Converter 的 write() 方法序列化对象到响应体

HttpMessageConverter的初始化

在WebMvcAutoConfiguration自动装配类中初始化EnableWebMvcConfiguration类,该类是WebMvcConfigurationSupport的子类。getMessageConverters()方法获取所有的消息转换器。

	protected final List<HttpMessageConverter<?>> getMessageConverters() {
		if (this.messageConverters == null) {//最开始是空的
			this.messageConverters = new ArrayList<>();
           //加载自定义的消息转换器
			configureMessageConverters(this.messageConverters);
			if (this.messageConverters.isEmpty()) {//如果未配置
              //加载默认的MessageConverter
				addDefaultHttpMessageConverters(this.messageConverters);
			}
           //添加扩展MessageConverters ,也是通过Configurer
			extendMessageConverters(this.messageConverters);
		}
		return this.messageConverters;
	}

configureMessageConverters会调用所有的WebMvcConfigurer加载定义的converter

	protected void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
       //这里的configurers是WebMvcConfigurerComposite实例
		this.configurers.configureMessageConverters(converters);
	}

WebMvcConfigurerComposite#configureMessageConverters

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

在WebMvcAutoConfiguration类中WebMvcAutoConfigurationAdapter就是一个默认的Configurer。其使用HttpMessageConverters来进行converter的初始化。

HttpMessageConverters的初始化在HttpMessageConvertersAutoConfiguration类中进行,首先会从beanFacotry中获取所有已经定义的HttpMessageConverter类型的bean,然后在获取所有的默认

HttpMessageConverters构造函数

	public HttpMessageConverters(boolean addDefaultConverters, Collection<HttpMessageConverter<?>> converters) {
		List<HttpMessageConverter<?>> combined = getCombinedConverters(converters,
				addDefaultConverters ? getDefaultConverters() : Collections.emptyList());
		combined = postProcessConverters(combined);
		this.converters = Collections.unmodifiableList(combined);
	}

第一步在HttpMessageConvertersAutoConfiguration类中默认定义了一个StringHttpMessageConverter,还会条件装配@ConditionalOnBean(ObjectMapper.class)一个MappingJackson2HttpMessageConverter类型的json转换其。这样在容器中默认有两个消息转换器。

第二部加载默认的消息转换器,这里入参addDefaultConverters是true,通过getDefaultConverters()方法加载。

	private List<HttpMessageConverter<?>> getDefaultConverters() {
		List<HttpMessageConverter<?>> converters = new ArrayList<>();
		if (ClassUtils.isPresent("org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport",
				null)) {
			converters.addAll(new WebMvcConfigurationSupport() {

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

			}.defaultMessageConverters());
		}
		else {
			converters.addAll(new RestTemplate().getMessageConverters());
		}
		reorderXmlConvertersToEnd(converters);
		return converters;
	}

这里会走if通过WebMvcConfigurationSupport.defaultMessageConverters()方法加载默认的消息解析器。

	protected final void addDefaultHttpMessageConverters(List<HttpMessageConverter<?>> messageConverters) {
		messageConverters.add(new ByteArrayHttpMessageConverter());
		messageConverters.add(new StringHttpMessageConverter());
		messageConverters.add(new ResourceHttpMessageConverter());
		messageConverters.add(new ResourceRegionHttpMessageConverter());
		if (!shouldIgnoreXml) {
			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 (!shouldIgnoreXml) {
			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 (kotlinSerializationJsonPresent) {
			messageConverters.add(new KotlinSerializationJsonHttpMessageConverter());
		}
		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()));
		}
	}

每个消息转换器类处理不同类型的报文体类型,这和报文的MediType相关。通过HttpMessageConverter.canRead( MediaType mediaType)来判断当前converter是否支持当前请求报文。

自定义HttpMessageConverter

如果默认内置的converter不能满足要求,可以进行自定义converter。如请求的body存在转码或加密,或有固定头信息,不是标准的json格式,则可以继承MappingJackson2HttpMessageConverter 类,然后重写对应的read和write方法。

自定义的converter可以通过WebMvcConfigurer加载到容器中。

@Configuration
public class WebConfiguration implements WebMvcConfigurer {

	@Override
	public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
		converters.add(new MyMappingJackson2HttpMessageConverter());	
	}
}

也可以通过将自定义的MessageConverter注册未一个bean来让容器自动加载。所有的MessageConverter是存放在一个list中,按converter的注册顺序进行加载。这里就有个小问题,如果converters列表有多个可以处理相同的的请求类型,只会使用第一个匹配到的converter,则可以通过list指定下标形式来设置自定义converter优先级。


网站公告

今日签到

点亮在社区的每一天
去签到