[转]@ResponseBody和Json序列化处理流程

本文详细探讨了在Spring框架中使用`@ResponseBody`注解的处理流程,以及Json序列化的过程。首先介绍`@ResponseBody`如何作用于处理器方法,然后讲解Spring如何遍历HttpMessageConverter列表来找到合适的转换器进行响应数据的处理。特别提到了`MappingJackson2HttpMessageConverter`类,它是用于Json序列化和反序列化的关键,通过ObjectMapper进行实际的转换操作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

原文地址:http://my.oschina.net/huanger/blog/373632

@ResponseBody的处理

  1. RequestMappingHandlerAdapter对handler进行调用,ServletInvocableHandlerMethod执行实际的调用.调用之后会使用HandlerMethodReturnValueHandlerComposite对返回值进行处理.

  2. HandlerMethodReturnValueHandlerComposite中其实包含了一个HandlerMethodReturnValueHandler的列表,遍历列表找出支持当前返回类型的(supportsReturnType),然后调用其handleReturnValue方法.

  3. RequestResponseBodyMethodProcessor就是上述列表中的一个HandlerMethodReturnValueHandler,其supportsReturnType如下:

    @Override
    public boolean supportsReturnType(MethodParameter returnType) {
        return ((AnnotationUtils.findAnnotation(returnType.getContainingClass(), ResponseBody.class) != null) ||
                (returnType.getMethodAnnotation(ResponseBody.class) != null));
    }
    

    这里出现了我们期待已久的@ResponseBody.

Json的序列化

 

  1. RequestResponseBodyMethodProcessor包含了从RequestMappingHandlerAdapter传过来的HttpMessageConverter列表.其handleReturnValue方法逻辑为:遍历HttpMessageConverter列表列表,找到可用的HttpMessageConverter(通过canWrite方法),然后调用HttpMessageConverterwrite的方法,对handler的结果进行处理.
  2. MappingJackson2HttpMessageConverter是第上一步列表中的一个HttpMessageConverter子类,负责json的序列化和反序列化.其中的序列化方法就是调用ObjectMapper进行处理的.
### SpringBoot 中 `@ResponseBody` 注解的序列化及其工作原理 在 Spring Boot 中,`@ResponseBody` 是一个重要的注解,主要用于将控制器中的方法返回值直接作为 HTTP 响应的内容发送给客户端。这个过程通常会涉及对象JSON 或 XML 的序列化操作。 #### 1. **`@ResponseBody` 的作用** 当在一个方法上标注了 `@ResponseBody` 后,Spring MVC 不再将该方法的返回值视为视图名称(View Name),而是将其直接写入 HTTP 响应体中[^2]。这意味着如果返回的是一个 Java 对象,则需要先将其换为适合传输的格式(通常是 JSON)。这种换由消息换器(Message Converter)负责完成。 #### 2. **序列化机制** 对于大多数现代 Web 应用程序来说,默认情况下,Spring Boot 配置好了 Jackson 来处理 JSON 数据的序列化序列化任务。因此当你标记某个 Controller 方法使用了 `@ResponseBody` 并且返回了一个 POJO 类型时,实际上就是调用了 Jackson 的 ObjectMapper 实例把那个对象成了 JSON 字符串[^4]。 例如: ```java @RestController @RequestMapping("/api/user") public class UserController { @GetMapping("/{id}") public ResponseEntity<User> getUserById(@PathVariable Long id){ User user = userService.findById(id); return new ResponseEntity<>(user, HttpStatus.OK); } } ``` 在这个例子中,`getUserById()` 函数被加了 `@RestController` (它是 `@Controller` 加上全局默认的 `@ResponseBody`) ,所以它的结果会被自动尝试换成 JSON 格式的字符串并通过网络传送给前端应用。 #### 3. **内部工作机制** 整个流程可以分解如下几个阶段: - 请求到达服务器端后,经过 Dispatcher Servlet 处理分发至相应的 Handler Mapping; - 接着找到匹配的具体 handler 执行业务逻辑得到最终的结果数据; - 此刻如果有 `@ResponseBody` 存在的话,就会触发 Message Converters 进行实际的数据格式换动作; - 默认启用 JsonConverter (基于 jackson-databind 库实现),它会依据目标类型决定采用何种方式渲染输出内容; - 最终生成好的文本流经 HttpServletResponse 输出回去给浏览器或者其它消费者接收解读[^3]. 总结起来讲,`@ResponseBody` 主要借助内置的消息换组件实现了从 Java Bean 到指定媒体类型的映射变换功能,从而简化开发者编写 RESTful APIs 的难度的同时也提高了系统的可维护性扩展能力。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值