SpringMVC HttpMessageConverter消息转换器

作者:这堆干货有点猛 时间:2023-01-13 04:57:45 

Spring MVC 为开发者提供了方便的开发方式和丰富的功能。其中,HttpMessageConverter 是Spring MVC中非常重要的一个组件,它负责将客户端提交的请求数据(如JSON、XML等)转换为Java对象,同时也负责将 Java 对象转换为客户端需要的数据格式。本文将 从HttpMessageConverter 的作用和源码实现两个方面进行讲解。

作用

在 Spring MVC 中,客户端通过发送请求,向服务器端请求数据,服务器端接收到请求后需要将请求参数转换为 Java 对象进行处理,这就需要使用 HttpMessageConverter,它主要负责将HTTP请求中的请求参数转换为 Java 对象或将 Java 对象转换为 HTTP 响应中的内容。

HttpMessageConverter 是一个接口,Spring MVC 中有很多实现类,用于处理不同的数据类型。常见的 HttpMessageConverter 实现类有:

  • ByteArrayHttpMessageConverter:用于处理字节数组类型的数据。

  • StringHttpMessageConverter:用于处理字符串类型的数据。

  • FormHttpMessageConverter:用于处理表单类型的数据。

  • MappingJackson2HttpMessageConverter:用于处理JSON类型的数据。

  • Jaxb2RootElementHttpMessageConverter:用于处理XML类型的数据。

HttpMessageConverter 接口定义了两个方法:canRead() 和 canWrite(),分别用于判断当前的 HttpMessageConverter 是否可以读取或写入指定的数据类型。

  • 如果 canRead() 方法返回 true,那么当前的 HttpMessageConverter 就可以将 HTTP 请求中的请求参数转换为 Java 对象;

  • 如果 canWrite() 方法返回 true,那么当前的 HttpMessageConverter 就可以将Java对象转换为 HTTP 响应中的内容。

源码实现

下面以 MappingJackson2HttpMessageConverter 为例,讲解 HttpMessageConverter 的源码实现。

MappingJackson2HttpMessageConverter 是一个用于处理 JSON 类型数据的 HttpMessageConverter 实现类。它继承了 AbstractJackson2HttpMessageConverter 类,并实现了 HttpMessageConverter 接口。

canRead()

@Override
public boolean canRead(Class<?> clazz, @Nullable MediaType mediaType) {
   JavaType javaType = getJavaType(clazz, null);
   return (this.objectMapper.canDeserialize(javaType) && canRead(mediaType));
}

方法首先调用了 getJavaType() 方法获取 JavaType 对象

然后判断当前的 ObjectMapper 是否可以反序列化 JavaType 对象。

如果当前的 HttpMessageConverter 可以反序列化 JavaType 对象并且当前的 MediaType 可以被处理,那么就返回 true,否则返回 false。

canWrite()

@Override
public boolean canWrite(Class<?> clazz, @Nullable MediaType mediaType) {
   return (this.objectMapper.canSerialize(clazz) && canWrite(mediaType));
}

方法首先判断当前的 ObjectMapper 是否可以序列化指定的 Java 对象

如果可以序列化并且当前的 MediaType 可以被处理,那么就返回 true,否则返回 false。

read()

@Override
public T readInternal(Class<? extends T> clazz, HttpInputMessage inputMessage)throws IOException, HttpMessageNotReadableException {
   JavaType javaType = getJavaType(clazz, null);
   try {
       return this.objectMapper.readValue(inputMessage.getBody(), javaType);
   }
   catch (JsonProcessingException ex) {
       throw new HttpMessageNotReadableException("JSON parse error: " + ex.getMessage(), ex, inputMessage);
   }
}

方法首先调用 getJavaType() 方法获取 JavaType 对象

然后通过 ObjectMapper 的 readValue() 方法将HTTP请求中的请求参数转换为 Java 对象。

如果在转换的过程中出现了异常,那么就抛出 HttpMessageNotReadableException 异常。

write()

@Override
protected void writeInternal(T t, HttpOutputMessage outputMessage) throws IOException, HttpMessageNotWritableException {
   try {
       JsonEncoding encoding = getJsonEncoding(outputMessage.getHeaders().getContentType());
       JsonGenerator jsonGenerator = this.objectMapper.getFactory().createGenerator(outputMessage.getBody(), encoding);
       writePrefix(jsonGenerator, t);
       this.objectMapper.writeValue(jsonGenerator, t);
       writeSuffix(jsonGenerator, t);
       jsonGenerator.flush();
   }
   catch (JsonProcessingException ex) {
       throw new HttpMessageNotWritableException("Could not write JSON: " + ex.getMessage(), ex);
   }
}

方法首先通过 getJsonEncoding() 方法获取JSON编码格式

然后通过 ObjectMapper 的 writeValue() 方法将 Java 对象转换为 JSON 格式的字符串

并将结果输出到 HTTP 响应中。如果在转换的过程中出现了异常,那么就抛出 HttpMessageNotWritableException 异常。

异同

大家是不是觉得与前面文章所讲的 MethodArgumentResolver 很相似,都能够将请求参数转换为Java对象,但它们的作用和处理流程不同:

  • HttpMessageConverter 会在请求到达 Controller 方法之前进行处理,这意味着它只处理请求参数,而不管 Controller 方法的签名和参数类型;

  • MethodArgumentResolver 会在 Controller 方法执行之前进行处理,它的作用是将请求参数按照 Controller 方法的参数顺序进行解析,并将解析结果映射到 Controller 方法的参数上,然后再调用 Controller 方法

来源:https://blog.csdn.net/wizard_hu/article/details/129873243

标签:SpringMVC,HttpMessageConverter,消息转换器
0
投稿

猜你喜欢

  • java集合求和最大值最小值示例分享

    2022-10-11 04:15:10
  • 详解Spring Boot最核心的27个注解,你了解多少?

    2023-11-20 06:23:35
  • 不使用Math.random方法生成随机数(随机数生成器)

    2021-11-28 05:08:46
  • 深入理解ThreadLocal工作原理及使用示例

    2022-02-27 19:24:14
  • Java ArrayList与LinkedList及HashMap容器的用法区别

    2022-07-07 00:53:25
  • Mybatis执行流程、缓存原理及相关面试题汇总

    2022-10-15 23:30:28
  • java synchronized的用法及原理详解

    2023-07-17 04:42:54
  • 关于@Autowired注解和静态方法及new的关系

    2021-07-16 13:08:06
  • Java异常处理之try...catch...finally详解

    2023-09-17 05:38:24
  • Java switch关键字原理及用法详解

    2023-04-16 23:29:49
  • Java进程cpu占用过高问题解决

    2021-08-09 00:16:59
  • Android ExpandableListView双层嵌套实现三级树形菜单

    2023-11-05 23:04:55
  • 一篇文章带你了解Java 中序列化与反序列化

    2021-11-27 21:19:27
  • Java选择排序和垃圾回收机制详情

    2023-10-23 16:53:38
  • Android基于OpenCV实现图像修复

    2022-04-28 00:51:39
  • Java DOM4J方式生成XML的方法

    2022-07-19 02:32:42
  • SpringBoot项目修改访问端口和访问路径的方法

    2023-07-13 11:23:35
  • Spark SQL关于性能调优选项详解

    2021-11-05 22:32:56
  • C#基于简单工厂模式实现的计算器功能示例

    2023-06-19 12:40:10
  • C#特性 匿名类型与隐式类型局部变量使用介绍

    2023-09-29 12:42:50
  • asp之家 软件编程 m.aspxhome.com