HttpMessageConverter(消息转换器)

news2024/11/24 2:50:13

文章目录

  • 1. HttpMessageConverter 的概述
  • 2. `HttpMessageConverter`的使用
    • 2.1 `@RequestBody`请求体注解
    • 2.2 `RequestEntity`请求封装对象
    • 2.3 `@ResponseBody`响应体注解
    • 2.4 `ResponseEntity`响应封装对象
  • 3. 自定义消息转换器
    • 3.1 步骤一:自定义消息转化器
      • 3.11 `FastJson、Gson` 等组件自带常用`json`消息转换器
      • 3.12 实现`HttpMessageConverter`接口
      • 3.13 继承`AbstractHttpMessageConverter`类
    • 3.2 步骤二:加载自定义消息转换器
      • 3.21 实现`WebMvcConfigurer#configureMessageConverters`接口方法
      • 3.22 实现`WebMvcConfigurer#extendMessageConverters`接口方法
  • 4. 推荐博客

1. HttpMessageConverter 的概述

我们先看HttpMessageConverter的介绍

/**
 * Strategy interface for converting from and to HTTP requests and responses.
 */
public interface HttpMessageConverter<T> {

简单说就是 HTTP request (请求)和response (响应)的转换器。该接口有只有5个方法,简单来说就是获取支持的 MediaType(application/json之类),接收到请求时判断是否能读(canRead),能读则读(read);返回结果时判断是否能写(canWrite),能写则写(write)

HttpMessageConverter 是 Spring 框架中的一个核心组件,用于在 HTTP 请求和响应之间进行消息转换。它的主要作用是将请求和响应的数据从 Java 对象转换为 HTTP 协议所需的格式,或者将 HTTP 协议中的数据转换为 Java 对象。

在 Spring MVC 中,当客户端发送一个 HTTP 请求时,Spring MVC 会根据请求中的内容类型(Content-Type)选择合适的 HttpMessageConverter 来处理请求体的数据。同样地,当服务器返回一个 HTTP 响应时,Spring MVC 会根据请求中的 Accept 头部信息选择合适的 HttpMessageConverter 来处理响应体的数据。
在这里插入图片描述

2. HttpMessageConverter的使用

HttpMessageConverter提供了两个注解和两个类型:@RequestBody、@ResponseBody、RequestEntity、ResponseEntity

特别说明一下RestController这个注解是一个复合注解,包含了@ResponseBody注解的

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Controller
@ResponseBody
public @interface RestController {

2.1 @RequestBody请求体注解

@RequestBody注解:在控制器方法参数上使用@RequestBody注解,表示将HTTP请求体转换为Java对象。它通常与POSTPUT等HTTP方法一起使用,用于接收客户端发送的数据。Spring框架会根据请求的Content-Type选择适当的HttpMessageConverter进行请求体的解析,并将解析后的Java对象作为方法参数传递给控制器方法。

示例代码片段:

@PostMapping("/example")
public void handleRequest(@RequestBody ExampleRequest requestBody) {
    // 处理请求体数据
}

2.2 RequestEntity请求封装对象

RequestEntity类型:RequestEntity是一个泛型类,用于表示完整的HTTP请求实体,包括请求头、请求体、请求方法和URI等信息。可以在方法参数或返回值中使用RequestEntity类型来获取或定义更多请求信息,例如自定义请求头、响应状态等。它提供了更灵活的方式来处理HTTP请求,允许对请求进行更细粒度的控制。

示例代码片段:

@PostMapping("/example")
public ResponseEntity<String> handleRequest(RequestEntity<ExampleRequest> requestEntity) {
    ExampleRequest requestBody = requestEntity.getBody();
    HttpHeaders requestHeaders = requestEntity.getHeaders();
    
    // 处理请求体和请求头信息
    // 构建响应实体并返回
}

2.3 @ResponseBody响应体注解

@ResponseBody是Spring框架中的一个注解,用于将方法的返回值转换为HTTP响应体。

当在控制器方法上使用@ResponseBody注解时,Spring会通过适当的HttpMessageConverter将方法的返回值转换为HTTP响应体,并设置响应的Content-Type头部。

使用@ResponseBody注解可以方便地将Java对象、字符串等转换为HTTP响应体,常用于返回JSON数据或纯文本数据给客户端。

  • 使用的时候首先引入jackjson
<dependency>
  <groupId>com.fasterxml.jackson.core</groupId>
  <artifactId>jackson-databind</artifactId>
  <version>2.13.4.2</version>
</dependency>
  • 然后在SpringMVC的核心配置文件中开启mvc的注解驱动

此时在HandlerAdaptor中会自动装配一个消息转换器:MappingJackson2HttpMessageConverter,可以将响应到浏览器的Java对象转换为Json格式的字符串

示例代码片段:

@GetMapping("/example")
@ResponseBody
public ExampleResponse getExample() {
    ExampleResponse response = new ExampleResponse();
    response.setMessage("Hello, World!");
    return response;
}

2.4 ResponseEntity响应封装对象

ResponseEntity是Spring框架中的一个泛型类,用于表示完整的HTTP响应实体,包括响应头、响应体和响应状态等信息。

通过使用ResponseEntity,我们可以更灵活地构建和定制HTTP响应,包括设置自定义的响应状态码、响应头部和响应体。它允许在控制器方法中明确指定特定的HTTP响应细节,并能够返回不同类型的响应内容(例如JSON、字符串、字节数组等)。

示例代码片段:

@GetMapping("/example")
public ResponseEntity<String> getExample() {
    String responseBody = "Hello, World!";
    HttpHeaders responseHeaders = new HttpHeaders();
    responseHeaders.setContentType(MediaType.TEXT_PLAIN);
    HttpStatus status = HttpStatus.OK;
    
    return new ResponseEntity<>(responseBody, responseHeaders, status);
}

在上述示例中,getExample()方法返回一个ResponseEntity<String>对象,它将字符串"Hello, World!“作为响应体,并设置了Content-Type头部为"text/plain”,状态码为200 OK。

ResponseEntity还提供其他方法来获取和设置响应实体的各个部分,例如:getBody()获取响应体,getHeaders()获取响应头部,getStatusCode()获取响应状态码等。

3. 自定义消息转换器

3.1 步骤一:自定义消息转化器

3.11 FastJson、Gson 等组件自带常用json消息转换器

3.12 实现HttpMessageConverter接口

package com.xmc.hello.config;

import cn.hutool.core.collection.ListUtil;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpInputMessage;
import org.springframework.http.HttpOutputMessage;
import org.springframework.http.MediaType;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter;
import org.springframework.util.StreamUtils;

import java.io.IOException;
import java.nio.charset.Charset;
import java.util.List;

/**
 * 自定义消息转化器(实现HttpMessageConverter)
 *
 * @author xmc
 */
public class MyHttpMessageConverter implements HttpMessageConverter<Object> {

    private MappingJackson2HttpMessageConverter defaultConverter;

    public MyHttpMessageConverter() {
        this.defaultConverter = new MappingJackson2HttpMessageConverter();
    }

    @Override
    public boolean canRead(Class<?> clazz, MediaType mediaType) {
        // 在此处指定支持的类和媒体类型
        return true;
    }

    @Override
    public boolean canWrite(Class<?> clazz, MediaType mediaType) {
        // 在此处指定支持的类和媒体类型
        return true;
    }

    @Override
    public List<MediaType> getSupportedMediaTypes() {
        // 指定支持的媒体类型
        return ListUtil.of(MediaType.APPLICATION_JSON);
    }

    @Override
    public Object read(Class<?> clazz, HttpInputMessage inputMessage) throws IOException {
        // 自定义读取请求消息的逻辑
        String requestBody = StreamUtils.copyToString(inputMessage.getBody(), Charset.defaultCharset());
        // 在这里进行转换操作,并返回相应的对象
        return null;
    }

    @Override
    public void write(Object o, MediaType contentType, HttpOutputMessage outputMessage) throws IOException {
        // 自定义写入响应消息的逻辑
        HttpHeaders headers = outputMessage.getHeaders();
        // 设置响应头部信息
        headers.setContentType(contentType);
        // 在这里进行转换操作,并将结果写入输出流
    }

    // 其他未覆盖的方法可以继续实现,根据需要进行自定义
}

在上述示例中,我们创建了一个名为 MyHttpMessageConverter 的类,它实现了 HttpMessageConverter<Object> 接口。根据需要,可以设置支持的类和媒体类型,以及自定义读取请求消息和写入响应消息的逻辑。

3.13 继承AbstractHttpMessageConverter

package com.xmc.hello.config;

import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpInputMessage;
import org.springframework.http.HttpOutputMessage;
import org.springframework.http.MediaType;
import org.springframework.http.converter.AbstractHttpMessageConverter;
import org.springframework.util.StreamUtils;

import java.io.IOException;
import java.nio.charset.Charset;

/**
 * 自定义消息转化器(实现HttpMessageConverter)
 *
 * @author xmc
 */
public class CustomHttpMessageConverter extends AbstractHttpMessageConverter<Object> {
    public CustomHttpMessageConverter() {
        super(MediaType.APPLICATION_JSON);
    }

    @Override
    protected boolean supports(Class<?> clazz) {
        // 在此处指定支持的类
        return true;
    }

    @Override
    protected Object readInternal(Class<? extends Object> clazz, HttpInputMessage inputMessage)
            throws IOException {
        // 自定义读取请求消息的逻辑
        String requestBody = StreamUtils.copyToString(inputMessage.getBody(), Charset.defaultCharset());
        // 在这里进行转换操作,并返回相应的对象
        return null;
    }

    @Override
    protected void writeInternal(Object o, HttpOutputMessage outputMessage) throws IOException {
        // 自定义写入响应消息的逻辑
        HttpHeaders headers = outputMessage.getHeaders();
        // 设置响应头部信息
        headers.setContentType(MediaType.APPLICATION_JSON);
        // 在这里进行转换操作,并将结果写入输出流
    }
}

在上述示例中,我们创建了一个名为CustomHttpMessageConverter 的类,它继承了AbstractHttpMessageConverter<Object> 抽象类。通过调用父类的构造函数并传入媒体类型,我们指定了该自定义消息转换器支持的媒体类型。

然后,我们重写了 supports 方法来指定支持的类。在 readInternal 方法中,您可以实现自定义的请求消息读取逻辑,并在 writeInternal 方法中实现自定义的响应消息写入逻辑。

3.2 步骤二:加载自定义消息转换器

3.21 实现WebMvcConfigurer#configureMessageConverters接口方法


package com.xmc.hello.config;

import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

import java.util.List;

/**
 * web资源配置
 * @author xmc
 */
@Configuration
public class WebMvcConfig implements WebMvcConfigurer {

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

}

在上述示例中,我们创建了一个名为WebMvcConfig的配置类,并重写了configureMessageConverters方法,将自定义的消息转换器直接添加到转换器列表中。

3.22 实现WebMvcConfigurer#extendMessageConverters接口方法


package com.xmc.hello.config;

import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

import java.util.List;

/**
 * web资源配置
 * @author xmc
 */
@Configuration
public class WebMvcConfig implements WebMvcConfigurer {

    /**
     * 实现 WebMvcConfigurer#extendMessageConverters 接口
     */
    @Override
    public void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
        // 注意加载顺序
        converters.add(0, new CustomHttpMessageConverter());
    }

}

在上述示例中,我们创建了一个名为WebMvcConfig的配置类,我们在extendMessageConverters方法中将CustomHttpMessageConverter转换器添加到默认的转换器列表中。

4. 推荐博客

HttpMessageConverter是这样转换数据的

【SpringMVC从入门到精通】06-HttpMessageConverter

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/893562.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

前馈神经网络正则化例子

直接看代码&#xff1a; import torch import numpy as np import random from IPython import display from matplotlib import pyplot as plt import torchvision import torchvision.transforms as transforms mnist_train torchvision.datasets.MNIST(root…

【云原生】kuberneter中Helm入门到实践

引言 helm是k8s的包管理工具&#xff0c;使用helm&#xff0c;可以使用更为简化和系统化的方式对k8s应用进行部署、升级。 helm是CNCF已毕业的项目&#xff0c;社区也是相当活跃的&#xff0c;在 https://artifacthub.io/ 上&#xff0c;能找到很多现成的helm chart&#xff…

轻松学会网络编程

目录 一、UDP 和 TCP 特点对比 1、有连接和无连接 2、可靠传输和不可靠传输 3、面向字节流和面向数据报 4、全双工和半双工 二、UDP 的 socket.api 1、DatagramSocket 2、DatagramPacket 回显服务器的实现 &#xff08;1&#xff09;服务器代码 &#xff08;2&#…

GaussDB数据库SQL系列-子查询

目录 一、前言 二、GaussDB SQL子查询表达式 1、EXISTS/NOT EXISTS 2、IN/NOT IN 3、ANY/SOME 4、ALL 三、GaussDB SQL子查询实验示例 1、创建实验表 2、EXISTS/NOT EXISTS示例 3、IN/NOT IN 示例 4、ANY/SOME 示例 5、ALL示例 四、注意事项及建议 五、小结 一、…

投资不识筹码峰,炒遍A股也枉然? | 如何用python计算筹码分布数据

你听说过股市上著名的丁蟹效应吗&#xff1f; 你知道丁蟹报仇点到为止&#xff0c;丁蟹报恩家破人亡吗&#xff1f; 你又是否曾在微信群中见过这些表情包&#xff1f; 01 大时代 不知道大家有没有看过《大时代》这部剧&#xff0c;看过的欢迎点我头像交流讨论。 剧中逆天强运…

Java:JVM虚拟机的三种模式

在JVM中有三种模式&#xff1a; 混合模式&#xff1a;解释器热点代码编译 编译模式&#xff1a;启动快&#xff0c;执行慢 解释模式&#xff1a;启动慢&#xff0c;执行快 使用 在我们的JVM虚拟机中一般默认的是混合模式 如上所示&#xff0c;我们可以看到后面有mixed&#xf…

【mysql异常】Specified key was too long; max key length is 1000 bytes

最近在创建数据库的时候&#xff0c;报错内容如下所示&#xff1a; Caused by: java.sql.SQLSyntaxErrorException: Specified key was too long; max key length is 1000 bytesat com.mysql.cj.jdbc.exceptions.SQLError.createSQLException(SQLError.java:120) ~[mysql-conn…

Vue中实现自动匹配搜索框内容 关键字高亮文字显示

实现效果如下: 1.首先需要给输入框进行双向绑定 2.拿到搜索的结果去渲染页面 将返回的结果和搜索的关键字进行比对 如果相同的 就变红 上代码 html部分 //输入框<div class"search"><div class"shuru"><input type"请输入要查询的…

软件测试报告有哪些测试内容?

软件测试报告可以包含以下测试内容&#xff1a; 1、功能测试&#xff1a;测试软件的基本功能是否实现&#xff0c;是否符合要求。 2、性能测试&#xff1a;测试软件的响应速度、并发能力、稳定性等性能指标。 3、界面测试&#xff1a;测试软件的用户界面是否友好、易于使用。 …

开车打电话买什么样的蓝牙好,分享几款通话性能最好的蓝牙耳机

随着时间的推移&#xff0c;如今的年轻人越来越倾向于使用骨传导耳机&#xff0c;因为他们都知道&#xff0c;骨传导耳机最大的优点就是带着很舒服的感觉&#xff0c;它不仅比普通的入耳式耳机更容易戴上&#xff0c;而且还比普通的入耳式耳机更安全&#xff0c;能有效地减少中…

try-with-resource语法使用

try-with-resources 是 Java 7 引入的一种语法结构&#xff0c;用于更方便地管理需要关闭的资源&#xff08;如 I/O 流、数据库连接等&#xff09;。它可以在代码块结束后自动关闭资源&#xff0c;无需显式调用 close() 方法&#xff0c;从而避免资源泄漏。 基本结构 try (Res…

opencv-python使用鼠标点击图片显示该点坐标和像素值IPM逆透视变换车道线

OpenCV的鼠标操作 实现获取像素点的功能主要基于OpenCV的内置函数cv2.setMouseCallback()&#xff0c;即鼠标事件回调 setMouseCallback(winname, onMouse,userdata0) winname: 接收鼠标事件的窗口名称 onMouse: 处理鼠标事件的回调函数指针 userdata: 传给回调函数的用户数据…

交流充电桩控制主板的优点

你是否曾经担心过充电桩可能会对你的电动车电池造成危害?让我们来探讨一下交流充电桩主板的优点&#xff0c;让你安心充电。 首先&#xff0c;交流充电桩主板采用了高安全性的电源设计&#xff0c;能够有效地保护电池免受电流、电压过高的危害&#xff0c;确保电池的安全使用。…

解决执行 spark.sql 时版本不兼容的一种方式

场景描述 hive 数据表的导入导出功能部分代码如下所示&#xff0c;使用 assemble 将 Java 程序和 spark 相关依赖一起打成 jar 包&#xff0c;最后 spark-submit 提交 jar 到集群执行。 public class SparkHiveApplication {public static void main(String[] args){long sta…

Dubbo—核心优势

一、快速易用 无论你是计划采用微服务架构开发一套全新的业务系统&#xff0c;还是准备将已有业务从单体架构迁移到微服务架构&#xff0c;Dubbo 框架都可以帮助到你。Dubbo 让微服务开发变得非常容易&#xff0c;它允许你选择多种编程语言、使用任意通信协议&#xff0c;并且…

什么是低价治理服务

当商品的销售价低于品牌要求的建议价时&#xff0c;就会被认为是低价销售&#xff0c;销售的主体是店铺&#xff0c;那店铺的运营方就成了低价的主导者&#xff0c;低价行为大部分品牌都会跟进&#xff0c;低价店铺的信息品牌也会去收集&#xff0c;因为只有掌握了低价链接、低…

什么是 脏写,脏读,幻读,不可重复读?怎样能解决这四种问题?

我们通过如下语句先创建一个 student 学生表。我就以对学生表的操作来解释什么是脏写&#xff0c;脏读&#xff0c;幻读&#xff0c;不可重复读 创建完成之后随便插入一条数据 1. 脏写&#xff1f; 对于两个事务 SessionA&#xff0c;SessionB&#xff0c;如果SessionA修改了另…

无公网IP,公网SSH远程访问家中的树莓派教程

文章目录 前言 如何通过 SSH 连接到树莓派步骤1. 在 Raspberry Pi 上启用 SSH步骤2. 查找树莓派的 IP 地址步骤3. SSH 到你的树莓派步骤 4. 在任何地点访问家中的树莓派4.1 安装 Cpolar内网穿透4.2 cpolar进行token认证4.3 配置cpolar服务开机自启动4.4 查看映射到公网的隧道地…

Timeplate Definition

timeplate定义描述单个tester cycle&#xff0c;并指定所有event edges被放置在cycle的位置。 必须在引用之前定义所有的timeplates。一个procedure必须有至少一个timeplate定义&#xff0c;所有的时钟必须在timeplate定义中进行定义&#xff0c;timeplate的定义有以下格式&am…

C++ STL关联式容器(详解)

STL关联式容器 C STL关联式容器是什么&#xff1f; 在《C STL容器》一节中讲到&#xff0c;C 容器大致分为 2 类&#xff0c;即序列式容器和关联式容器。其中&#xff0c;序列式容器&#xff08;包括 array、vector、list、deque 和 forward_list&#xff09;已经在前面章节中…