当前位置:首页 > 技术分析 > 正文内容

面试官问,你知道http请求怎么在你的项目中变成Java对象吗?

ruisui883个月前 (02-04)技术分析21

1.基本介绍

相信使用过Spring的同学都用过@RequestBody、@ResponseBody注解。当一个Http报文到达服务器,我们看到的是在Controller中直接得到了对应的对象,意味着浏览器和服务器通过交换原始文本进行通信,而这里其实就是HttpMessageConverter发挥着作用。

本文简单剖析该组件的原理,并且结合业务实际演示在开发中如何自定义完成特殊的操作

2.作用原理

如果一句话来描述HttpMessageConvert是什么,便是将Java对象与HttpInput/Output流做转换。


3.基本接口

public interface HttpMessageConverter<T> {

    /**
     * 该方法指定转换器可以读取的对象类型,即转换器可将请求信息转换为clazz类型的对象,
     * 同时指定支持的MIME类型(text/html、application/json等)。
     */
    boolean canRead(Class<?> clazz, MediaType mediaType);

    /**
     * 该方法指定转换器可以讲clazz类型的对象写到响应流当中,响应流支持的媒体类型在mediaType中定义
     * 
     */
    boolean canWrite(Class<?> clazz, MediaType mediaType);

    /**
     * 该方法返回当前转换器支持的媒体类型
     */
    List<MediaType> getSupportedMediaTypes();

    /**
     * 该方法将请求信息转换为T类型的对象
     */
    T read(Class<? extends T> clazz, HttpInputMessage inputMessage)
            throws IOException, HttpMessageNotReadableException;

    /**
     * 该方法将T类型的对象写到响应流当中,同事指定响应的媒体类型为contentType
     * 
     */
    void write(T t, MediaType contentType, HttpOutputMessage outputMessage)
            throws IOException, HttpMessageNotWritableException;

}

4.Spring内部封装

在四个主要接口的基础上,Spring还封装了一个抽象基类 AbstractHttpMessageConverter

其中主要增加了在何种情况下应该使用该HttpMessageConverter,包括编码、ContentType等等。

同时还有返回结果转换时候的ContentType,针对不同Converter会给前端不同的返回结果。

Spring为我们提供了一些通用的实现方法:

  • StringHttpMessageConverter
  • ByteArrayHttpMessageConverter
  • SourceHttpMessageConverter

这也就是为什么在Controller中直接返回一个String对象前端可以直接得到一个plain/text的返回内容。

5.选择策略

每个工程一定会有多个HttpMessageConverter,对于不同的注解将会用不同的MessageConverter去转换:

  • @RequestBody注解时: 根据Request对象header部分的Content-Type类型,逐一匹配合适的HttpMessageConverter来读取数据
  • @ResponseBody注解时: 根据Request对象header部分的Accept属性(逗号分隔),逐一按accept中的类型,去遍历找到能处理的HttpMessageConverter

在Spring内部有一个HttpMessageConverterList,上述过程中每次会去顺序遍历这个List,一旦找到匹配项,将不再向后查找

6.实际应用

在我们的业务工程中其实已经有对该部分的使用。

1) 第三方HttpMessageConverter

比如我们用的Json转换:MappingJackson2HttpMessageConverter

同理还有FastjsonHttpMessageConverter

2) 添加MessageConverter

@Configuration
public class WebMvcConfigurer extends WebMvcConfigurerAdapter {

    private static final String UTF8 = "UTF-8";

    private final SessionInterceptor sessionInterceptor;

    public WebMvcConfigurer(
            final SessionInterceptor sessionInterceptor) {
        this.sessionInterceptor = sessionInterceptor;
    }

    @Override
    public void addInterceptors(final InterceptorRegistry registry) {
        registry.addInterceptor(sessionInterceptor);
    }

    @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        super.configureMessageConverters(converters);
        final Charset utf8Charset = Charset.forName(UTF8);
        converters.add(new StringHttpMessageConverter(utf8Charset));
        final MappingJackson2HttpMessageConverter jackson2Converter = new MappingJackson2HttpMessageConverter();
        jackson2Converter.setDefaultCharset(utf8Charset);
        jackson2Converter.getObjectMapper().setSerializationInclusion(JsonInclude.Include.NON_NULL);
        SimpleModule module = new SimpleModule();
        module.addSerializer(String.class, new UpgradeInsecureRequestsStringSerializer());
        jackson2Converter.getObjectMapper().registerModule(module);
        converters.add(jackson2Converter);
    }
}

通过继承WebMvcConfigurerAdapter可以添加自定义的转换类型

注意自己自定义的转换会在系统自带转换之后

比如jackson中可以进一步自定义对objectMapper的详细设置

7.问题排查

如果以后在业务中会遇到转换json不对,或者string转换有问题的情况,那么可以尝试从HttpMessageConverter入手排查。

扫描二维码推送至手机访问。

版权声明:本文由ruisui88发布,如需转载请注明出处。

本文链接:http://www.ruisui88.com/post/1665.html

标签: objectmapper
分享给朋友:

“面试官问,你知道http请求怎么在你的项目中变成Java对象吗?” 的相关文章

基于Ubuntu的Linux Lite 6.0发行版正式发布

开发者 Jerry Bezencon 今天宣布,基于 Ubuntu 的 Linux Lite 6.0 发行版正式发布。本次更新代号为“Fluorite”,基于 Ubuntu 22.04 LTS (Jammy Jellyfish) 系统,它由长期支持的 Linux 5.15 LTS 内核系列驱动。下载...

云时代,Linux和容器为王,微软推出自己的发行版Azure Linux

看到今天的微软,其前CEO史蒂夫.鲍尔默肯定会后悔在15年前说过的一句话“Linux 是一种癌症(Linux is a cancer)”。 因为今天的微软不光靠Linux赚的钵满盆满,而且还发行了自己的Linux发行版Azure Linux。为了提高其云服务和容器化服务水平,其Azure公有云平台提...

「 VUE3 + TS + Vite 」父子组件间如何通信?

组件之间传值,大家都很熟悉,涉及到 VUE3 +TS 好多同学就无从下手了,所以分享这篇文章,希望看完后提起 VUE3+TS 能够不慌不忙。平时使用的函数如:ref、reactive、watch、computed 等需要先引入才能使用,但是本篇文章介绍的 defineProps、withDefaul...

代码管理-9-gitlab的使用和设置

gitlab使用1、外观设置完成后保存,返回登录页面查看关于注册,有些公司是不允许打开的,,有些人数非常多的公司就需要打开注册的功能,让人员自己注册,我们来给他特定的权限就可以,毕竟人非常多的时候还由我们来给她们注册就非常不现实了,工作量会很大2、自动注册3、组&用户&项目创建组设置组名称、描述等创...

js中数组filter方法的使用和实现

定义filter() 方法创建一个新数组, 其包含通过所提供函数实现的测试的所有元素。语法var newArray = arr.filter(callback(element[, index[, selfArr]])[, thisArg])参数callback循环数组每个元素时调用的回调函数。回调函...

vue.js 双向绑定如何理解,有什么好处!#云南小程序开发

Vue.js 的双向数据绑定是借助于 JavaScript 的一些特性,如对象的属性 getter 和 setter 以及 Vue 的依赖追踪系统实现的。简单来说,双向数据绑定就是数据与视图间的双向通信,也就是说数据的改变会马上反映到视图中,视图的改变也会立刻改变数据。具体来说,当你改变了数据时,视...