博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【Spring】HttpMessageConverter的作用及替换
阅读量:6201 次
发布时间:2019-06-21

本文共 4903 字,大约阅读时间需要 16 分钟。

相信使用过Spring的开发人员都用过@RequestBody、@ResponseBody注解,可以直接将输入解析成Json、将输出解析成Json,但HTTP 请求和响应是基于文本的,意味着浏览器和服务器通过交换原始文本进行通信,而这里其实就是HttpMessageConverter发挥着作用。

HttpMessageConverter

Http请求响应报文其实都是字符串,当请求报文到java程序会被封装为一个ServletInputStream流,开发人员再读取报文,响应报文则通过ServletOutputStream流,来输出响应报文。

从流中只能读取到原始的字符串报文,同样输出流也是。那么在报文到达SpringMVC / SpringBoot和从SpringMVC / SpringBoot出去,都存在一个字符串到java对象的转化问题。这一过程,在SpringMVC / SpringBoot中,是通过HttpMessageConverter来解决的。HttpMessageConverter接口源码:

public interface HttpMessageConverter
{ boolean canRead(Class
clazz, MediaType mediaType); boolean canWrite(Class
clazz, MediaType mediaType); List
getSupportedMediaTypes(); T read(Class
clazz, HttpInputMessage inputMessage) throws IOException, HttpMessageNotReadableException; void write(T t, MediaType contentType, HttpOutputMessage outputMessage) throws IOException, HttpMessageNotWritableException;}复制代码

下面以一例子来说明:

@RequestMapping("/test")@ResponseBodypublic String test(@RequestBody String param) {    return "param '" + param + "'";}复制代码

在请求进入test方法前,会根据@RequestBody注解选择对应的HttpMessageConverter实现类来将请求参数解析到param变量中,因为这里的参数是String类型的,所以这里是使用了StringHttpMessageConverter类,它的canRead()方法返回true,然后read()方法会从请求中读出请求参数,绑定到test()方法的param变量中。

同理当执行test方法后,由于返回值标识了@ResponseBody,SpringMVC / SpringBoot将使用StringHttpMessageConverter的write()方法,将结果作为String值写入响应报文,当然,此时canWrite()方法返回true。

借用下图简单描述整个过程:

00001.png

在Spring的处理过程中,一次请求报文和一次响应报文,分别被抽象为一个请求消息HttpInputMessage和一个响应消息HttpOutputMessage。

处理请求时,由合适的消息转换器将请求报文绑定为方法中的形参对象,在这里同一个对象就有可能出现多种不同的消息形式,如json、xml。同样响应请求也是同样道理。

在Spring中,针对不同的消息形式,有不同的HttpMessageConverter实现类来处理各种消息形式,至于各种消息解析实现的不同,则在不同的HttpMessageConverter实现类中。

替换@ResponseBody默认的HttpMessageConverter

这里使用SpringBoot演示例子,在SpringMVC / SpringBoot中@RequestBody这类注解默认使用的是jackson来解析json,看下面例子:

@Controller@RequestMapping("/user")public class UserController {    @RequestMapping("/testt")    @ResponseBody    public User testt() {        User user = new User("name", 18);        return user;    }}复制代码
public class User {    private String username;    private Integer age;        private Integer phone;        private String email;    public User(String username, Integer age) {    super();    this.username = username;    this.age = age;    }}复制代码

浏览器访问/user/testt返回如下:

00002.png

这就是使用jackson解析的结果,现在来改成使用fastjson解析对象,这里就是替换默认的HttpMessageConverter,就是将其改成使用FastJsonHttpMessageConverter来处理Java对象与HttpInputMessage/HttpOutputMessage间的转化。

首先新建一配置类来添加配置FastJsonHttpMessageConverter,Spring4.x开始推荐使用Java配置加注解的方式,也就是无xml文件,SpringBoot就更是了。

import com.alibaba.fastjson.serializer.SerializerFeature;import com.alibaba.fastjson.support.config.FastJsonConfig;import com.alibaba.fastjson.support.spring.FastJsonHttpMessageConverter;import org.springframework.boot.autoconfigure.web.HttpMessageConverters;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.http.converter.HttpMessageConverter;import java.nio.charset.Charset;@Configurationpublic class HttpMessageConverterConfig {    //引入Fastjson解析json,不使用默认的jackson    //必须在pom.xml引入fastjson的jar包,并且版必须大于1.2.10    @Bean    public HttpMessageConverters fastJsonHttpMessageConverters() {        //1、定义一个convert转换消息的对象        FastJsonHttpMessageConverter fastConverter = new FastJsonHttpMessageConverter();        //2、添加fastjson的配置信息        FastJsonConfig fastJsonConfig = new FastJsonConfig();        SerializerFeature[] serializerFeatures = new SerializerFeature[]{                //    输出key是包含双引号//                SerializerFeature.QuoteFieldNames,                //    是否输出为null的字段,若为null 则显示该字段//                SerializerFeature.WriteMapNullValue,                //    数值字段如果为null,则输出为0                SerializerFeature.WriteNullNumberAsZero,                //     List字段如果为null,输出为[],而非null                SerializerFeature.WriteNullListAsEmpty,                //    字符类型字段如果为null,输出为"",而非null                SerializerFeature.WriteNullStringAsEmpty,                //    Boolean字段如果为null,输出为false,而非null                SerializerFeature.WriteNullBooleanAsFalse,                //    Date的日期转换器                SerializerFeature.WriteDateUseDateFormat,                //    循环引用                SerializerFeature.DisableCircularReferenceDetect,        };        fastJsonConfig.setSerializerFeatures(serializerFeatures);        fastJsonConfig.setCharset(Charset.forName("UTF-8"));        //3、在convert中添加配置信息        fastConverter.setFastJsonConfig(fastJsonConfig);        //4、将convert添加到converters中        HttpMessageConverter
converter = fastConverter; return new HttpMessageConverters(converter); }}复制代码

这里将字符串类型的值如果是null就返回“”,数值类型的如果是null就返回0,重启应用,再次访问/user/testt接口,返回如下:

00003.png

可以看到此时null都转化成“”或0了。

转载自:https://www.jianshu.com/p/333ed5ee958d

转载于:https://juejin.im/post/5c85bd3c5188257ea64cedeb

你可能感兴趣的文章
Android应用程序消息处理机制(Looper、Handler)分析(4)
查看>>
C++ 类成员的构造和析构顺序
查看>>
将String转化成Stream,将Stream转换成String
查看>>
java路径Java开发中获得非Web项目的当前项目路径
查看>>
Google API设计指南-资源名称
查看>>
最全React技术栈技术资料汇总(收藏)
查看>>
【工具使用系列】关于 MATLAB 遗传算法与直接搜索工具箱,你需要知道的事
查看>>
flex 学习笔记 stage
查看>>
Kali-linux Arpspoof工具
查看>>
java中三个类别加载器的关系以及各自加载的类的范围
查看>>
PDF文档页面如何重新排版?
查看>>
基于http协议使用protobuf进行前后端交互
查看>>
python3 + Django + uwsgi + nginx 配置部署笔记
查看>>
UML设计一个电影票务销售系统(四)
查看>>
如何给VEEAM 7 分配角色权限
查看>>
AlphaGo Zero用它来调参?【高斯过程】到底有何过人之处?
查看>>
《Redis官方教程》Redis集群规范(二)
查看>>
MacOS必备软件
查看>>
卸载金蝶kis记账王的方法
查看>>
centos中系统启动故障排除
查看>>