当前位置: 首页 > news >正文

Spring MVC HandlerAdapter 的作用是什么? 为什么 DispatcherServlet 不直接调用 Controller 方法?

HandlerAdapter 在 Spring MVC 中扮演着**适配器(Adapter)**角色。它的核心作用是适配不同类型的处理器。

一、 HandlerAdapter 的核心作用:

  1. 适配不同类型的 Handler: Spring MVC 支持多种定义 Handler 的方式:

    • 目前版本中常用的:使用 @Controller 注解的类中带有 @RequestMapping 注解的方法 (HandlerMethod)。
    • 旧式的:实现 org.springframework.web.servlet.mvc.Controller 接口的类。
    • 旧式的:实现 org.springframework.web.HttpRequestHandler 接口的类。
    • 函数式:HandlerFunction (配合 RouterFunction 使用)。
    • 还有自定义 Handler 类型。
      这些不同类型的 Handler,其调用方式、参数解析方式、返回值处理方式都可能不同。DispatcherServlet 本身并不知道如何调用具体的 Handler 类型。
  2. 统一调用接口: HandlerAdapter 提供了一个统一的接口(handle 方法),DispatcherServlet 只需要调用这个统一的接口,而无需关心底层 Handler 的具体实现细节。HandlerAdapter 内部负责统一调用具体的 Handler 类型。

  3. 封装调用细节: HandlerAdapter 不仅仅是简单的调用 Handler 方法,它还负责处理调用过程中的复杂细节,特别是对于注解驱动的 HandlerMethod

    • 方法参数解析 (Argument Resolution): 利用 HandlerMethodArgumentResolver 解析 Controller 方法的各种参数(如 @RequestParam, @PathVariable, @RequestBody, Model, HttpServletRequest 等),从请求中提取数据、进行类型转换、数据绑定和校验。
    • 方法返回值处理 (Return Value Handling): 利用 HandlerMethodReturnValueHandler 处理 Controller 方法的各种返回值类型(如 String 视图名、ModelAndView@ResponseBody 对象、ResponseEntity 等),进行视图解析或写入响应体。

二、 为什么 DispatcherServlet 不直接调用 Controller 方法?

DispatcherServlet 不直接调用 Controller 方法,而是通过 HandlerAdapter 进行调用,主要是基于以下设计原则和考虑:

  1. 遵循开闭原则 (Open/Closed Principle):

    • 对扩展开放: 如果未来 Spring MVC 需要支持一种全新的 Handler 类型,只需要增加一个新的 HandlerAdapter 实现即可,而不需要修改 DispatcherServlet 的核心代码
    • 对修改关闭: DispatcherServlet 的核心调度逻辑保持不变,不用因为 Handler 类型的增多而变化。
  2. 保持 DispatcherServlet 的职责单一:

    • DispatcherServlet 的核心职责是请求的调度和协调,它负责整个请求处理流程的控制。如果让它直接处理各种 Handler 的调用细节(参数解析、返回值处理等),就会违反单一职责原则,使其变得过于庞大和复杂,难以维护。
  3. 解耦 (Decoupling):

    • DispatcherServlet 与具体的 Handler 实现解耦。DispatcherServlet 只依赖于 HandlerAdapter 接口,不依赖于任何具体的 Controller 或 Handler 实现类。这使得系统更加灵活,组件之间的依赖性更低。
  4. 提高可测试性:

    • 可以独立的测试 DispatcherServlet 的调度逻辑,也可以独立的测试不同的 HandlerAdapter 对特定 Handler 类型的适配和调用逻辑。
  5. 封装复杂性:

    • 调用一个注解驱动的 Controller 方法涉及到很多复杂的操作(反射调用、参数解析、数据绑定、类型转换、校验、返回值处理等)。将这些复杂的逻辑封装在专门的 HandlerAdapter(如 RequestMappingHandlerAdapter)内部,使得 DispatcherServlet 的代码保持简洁。

总结:

HandlerAdapter 的存在是 Spring MVC 框架灵活性和可扩展性的关键体现。它像一个万能插座转换器一样,允许 DispatcherServlet 这个“电源插座”能够驱动各种不同“插头”(Handler 类型)的“电器”。通过引入 HandlerAdapter 这一层,Spring MVC 实现了对不同处理器类型的统一调度,保持了 DispatcherServlet 的简洁和稳定,并遵循了良好的面向对象设计原则(如开闭原则、单一职责原则),使得框架易于扩展和维护。如果 DispatcherServlet 直接调用 Controller 方法,整个框架的设计将会变得非常脆弱和僵化。

相关文章:

  • YOLOv8融合CPA-Enhancer【提高恶略天气的退化图像检测】
  • oracle 锁的添加方式和死锁的解决
  • Yocto meta-toradex-security layer 创建独立数据分区
  • MongoDB副本集搭建与核心机制
  • 【回眸】香橙派Zero2(全志H616)初探
  • 2026届华为海思秋暑期IC实习秋招笔试真题(2025.04.23更新)
  • 函数的多种参数使用形式
  • 驱动开发系列53 - 一个OpenGL应用程序是如何调用到驱动厂商GL库的
  • 基于Python爬虫的音乐歌手的歌名和歌词信息爬取(可以输入歌手名字,然后爬取到该歌手的全部歌名和歌词信息)
  • Ubuntu主机上通过WiFi转有线为其他设备提供网络连接
  • 【蓝桥杯】产值调整
  • 基于大模型的结肠癌全病程预测与诊疗方案研究
  • Android插拔U盘导致黑屏问题排查
  • macOS 连接远程服务器的推荐方法和工具
  • Kingbase性能优化浅谈
  • 《深入理解计算机系统》阅读笔记之第一章 计算机系统漫游
  • SVN 右键不显示clean up的解决方法
  • java—11 Redis
  • vxe-table封装表头
  • 10天学会嵌入式技术之51单片机-day-7
  • 建设高标准农田主要目标是什么?有哪些安排?两部门有关负责人答问
  • 最高法:学校未及时发现并制止校园暴力行为,需承担侵权责任
  • 中国建设银行原党委委员、副行长章更生严重违纪违法被开除党籍
  • 国产手术机器人+5G技术,上海医生同一天远程为五地患者开刀
  • 医学泰斗客死他乡?AI小作文批量如何炮制?对话已被抓获的网络水军成员
  • 山西一国道塌陷致2死后续:地质雷达检测出10处道路病害