Spring MVC 中请求处理流程及核心组件解析

在 Spring MVC 中,请求从客户端发送到服务器后,需要经过一系列组件的处理才能最终到达具体的 Controller 方法。这个过程涉及多个核心组件和复杂的映射机制,下面详细解析其工作流程:

1. 核心组件与请求流程

Spring MVC 的请求处理流程主要涉及以下核心组件:

DispatcherServlet:前端控制器,接收所有 HTTP 请求,是整个请求处理的入口。

HandlerMapping:请求映射处理器,负责将请求 URL 映射到对应的 Handler(即 Controller 方法)。

HandlerAdapter:处理器适配器,将不同类型的 Handler 统一为可执行的接口。

Controller:具体的业务控制器,处理请求并返回 ModelAndView。

ViewResolver:视图解析器,将逻辑视图名解析为具体的 View 对象。

View:视图对象,负责将模型数据渲染到客户端。

请求处理流程

客户端请求 → DispatcherServlet → HandlerMapping → HandlerAdapter →Controller → ModelAndView → ViewResolver → View → 响应客户端

2. HandlerMapping 的请求映射机制

HandlerMapping 是实现 URL 到 Controller 方法映射的核心组件,Spring MVC 提供了多种实现:

RequestMappingHandlerMapping:处理 @RequestMapping 注解的映射(最常用)。

BeanNameUrlHandlerMapping:根据 Bean 名称映射请求(如 /hellohelloController Bean)。

SimpleUrlHandlerMapping:通过配置文件手动指定 URL 与 Handler 的映射关系。

基于注解的映射流程

当使用 @RequestMapping 或其派生注解(如 @GetMapping)时:

启动时扫描:Spring MVC 在启动时会扫描所有带有 @Controller@RestController 注解的 Bean。

解析方法映射:提取这些 Bean 中所有标注了 @RequestMapping 的方法,构建映射关系。

生成 HandlerMethod 对象:将每个映射方法封装为 HandlerMethod 对象,包含控制器实例、方法反射信息和请求匹配条件(如 URL、请求方法、请求参数等)。

注册到映射器:将 HandlerMethod 对象注册到 RequestMappingHandlerMapping 的内部映射表中。

3. DispatcherServlet 的请求分发

当接收到请求时,DispatcherServlet 会按以下步骤处理:

获取 HandlerExecutionChain

HandlerExecutionChain handler = getHandler(request);

DispatcherServlet 调用 HandlerMappinggetHandler(request) 方法,根据请求 URL 查找匹配的 Handler。

HandlerMapping 返回一个 HandlerExecutionChain 对象,包含 Handler(即 Controller 方法)和相关的拦截器(Interceptor)。

获取 HandlerAdapter

HandlerAdapter ha = getHandlerAdapter(handler.getHandler());

DispatcherServlet 根据 Handler 的类型,从已注册的 HandlerAdapter 中选择合适的适配器。

例如,RequestMappingHandlerAdapter 用于处理基于注解的 Controller 方法。

执行 Handler

ModelAndView mv = ha.handle(request, response, handler.getHandler());

HandlerAdapter 调用 Controller 方法,并传递请求参数,获取返回的 ModelAndView

4. 参数解析与返回值处理

参数解析

HandlerAdapter 通过 HandlerMethodArgumentResolver 解析请求参数:

内置解析器:Spring MVC 提供多种解析器,例如:

RequestParamMethodArgumentResolver:解析 @RequestParam 注解的参数。

PathVariableMethodArgumentResolver:解析路径变量(如 /users/{id})。

RequestBodyArgumentResolver:解析请求体(如 JSON、XML),需配合 @RequestBody 注解。

自定义解析器:可实现 HandlerMethodArgumentResolver 接口自定义参数解析逻辑。

返回值处理

HandlerAdapter 通过 HandlerMethodReturnValueHandler 处理返回值:

内置处理器:例如:

ViewNameMethodReturnValueHandler:处理返回视图名的方法。

ModelAndViewMethodReturnValueHandler:处理返回 ModelAndView 的方法。

ResponseBodyEmitterReturnValueHandler:处理异步返回值(如 ResponseEntity)。

@ResponseBody 注解:若方法标注了 @ResponseBody,则通过 HttpMessageConverter 将返回值序列化为响应体(如 JSON)。

5. 示例:从请求到 Controller 的完整流程

假设存在以下 Controller:

@RestController@RequestMapping("/api/users")public class UserController {@Autowiredprivate UserService userService;@GetMapping("/{id}")public UserDTO getUser(@PathVariable Long id) {return userService.getUserById(id);}}

请求流程

客户端发送请求 GET /api/users/123

DispatcherServlet 接收请求,调用 RequestMappingHandlerMapping

HandlerMapping 根据 URL /api/users/123 和请求方法 GET,匹配到 UserController.getUser() 方法。

DispatcherServlet 获取 HandlerAdapter(即 RequestMappingHandlerAdapter)。

HandlerAdapter 解析路径变量 id=123,并调用 UserController.getUser(123)

Controller 方法返回 UserDTO 对象。

由于 @RestController 隐含 @ResponseBody,HandlerAdapter 通过 HttpMessageConverterUserDTO 序列化为 JSON。

DispatcherServlet 将 JSON 响应返回给客户端。

6. 拦截器与异常处理

拦截器(Interceptor)

在请求到达 Controller 前后,可通过拦截器执行额外逻辑:

实现 HandlerInterceptor 接口

public class LoggingInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response,Object handler) {// 请求处理前执行(如日志记录、权限校验)return true; // 继续执行后续流程}@Overridepublic void postHandle(HttpServletRequest request,HttpServletResponse response,Object handler,ModelAndView modelAndView) {// 请求处理后、视图渲染前执行}@Overridepublic void afterCompletion(HttpServletRequest request,HttpServletResponse response,Object handler,Exception ex) {// 完成请求处理(如资源清理)}}

注册拦截器

@Configurationpublic class WebConfig implements WebMvcConfigurer {@Overridepublic void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new LoggingInterceptor()).addPathPatterns("/api/**"); // 拦截所有 API 请求}}
异常处理

通过 @ControllerAdvice@ExceptionHandler 统一处理异常:

@ControllerAdvicepublic class GlobalExceptionHandler {@ExceptionHandler(NotFoundException.class)public ResponseEntity<String> handleNotFoundException(NotFoundException ex) {return ResponseEntity.status(HttpStatus.NOT_FOUND).body(ex.getMessage());}}

总结

Spring MVC 通过 HandlerMapping 实现 URL 到 Controller 方法的映射,通过 HandlerAdapter 统一调用不同类型的 Handler,并借助 参数解析器返回值处理器 完成请求参数和响应数据的转换。整个过程由 DispatcherServlet 协调,配合拦截器和异常处理机制,形成了一个完整的请求处理体系。理解这些机制,有助于更高效地开发和调试 Spring MVC 应用。

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

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

相关文章

RISC-V 开发板 MUSE Pi Pro V2D图像加速器测试,踩坑介绍

视频讲解&#xff1a; RISC-V 开发板 MUSE Pi Pro V2D图像加速器测试&#xff0c;踩坑介绍 今天测试下V2D&#xff0c;这是K1特有的硬件级别的2D图像加速器&#xff0c;参考如下文档&#xff0c;但文档中描述的部分有不少问题&#xff0c;后面会讲下 https://bianbu-linux.spa…

hbase shell的常用命令

一、hbase shell的基础命令 # 版本号查看 [rootTest-Hadoop-NN-01 hbase]$ ./bin/hbase version HBase 2.4.0 Source code repository git://apurtell-ltm.internal.salesforce.com/Users/apurtell/src/hbase revision282ab70012ae843af54a6779543ff20acbcbb629# 客户端登录 […

深入解析Python中的Vector2d类:从基础实现到特殊方法的应用

引言 在Python面向对象编程中&#xff0c;特殊方法&#xff08;或称魔术方法&#xff09;是实现对象丰富行为的关键。本文将以Vector2d类为例&#xff0c;详细讲解如何通过特殊方法为自定义类添加多种表示形式和操作能力。 Vector2d类的基本行为 Vector2d类是一个二维向量类…

Zookeeper入门(三)

Zookeeper Java 客户端 项目构建 ookeeper 官方的客户端没有和服务端代码分离&#xff0c;他们为同一个jar 文件&#xff0c;所以我们直接引入 zookeeper的maven即可&#xff0c; 这里版本请保持与服务端版本一致&#xff0c;不然会有很多兼容性的问题 1 <dependency>…

Redis的主从架构

主从模式 全量同步 首先主从同步过程第一步 会先比较replication id 判断是否是第一次同步假设为第一次同步 那么就会 启动bgsave异步生成RDB 同时fork子进程记录生成期间的新数据发送RDB给从节点 清空本地数据写入RDB 增量同步 对比ReplicationID不同因此选择增量同步在Rep…

新电脑软件配置二:安装python,git, pycharm

安装python 地址 https://www.python.org/downloads/ 不是很懂为什么这么多版本 安装windows64位的 这里我是凭自己感觉装的了 然后cmd输入命令没有生效&#xff0c;先重启下&#xff1f; 重启之后再次验证 环境是成功的 之前是输入的python -version 命令输入错误 安装pyc…

docker 学习记录

docker pull nginx docker 将本地nginx快照保存到当前文件夹下 docker save -o nginx.tar nginx:latestdocker 将本地nginx 加载 docker load -i nginx.tar docker运行nginx在80端口 docker run --name dnginx -p 80:80 -d nginxredis启动 docker run --name mr -p 6379:6379 -…

什么是私有IP地址?如何判断是不是私有ip地址

在互联网的世界中&#xff0c;IP地址是设备之间通信的基础标识。无论是浏览网页、发送邮件还是在线游戏&#xff0c;IP地址都扮演着至关重要的角色。然而&#xff0c;并非所有的IP地址都是公开的&#xff0c;有些IP地址被保留用于内部网络&#xff0c;这就是我们所说的私有IP地…

功能安全管理

一、功能安全整体管理 1、功能安全文化&#xff0c;良好的功能安全文化包括&#xff1a; 1&#xff09; 在公司层面&#xff0c;有清晰的组织架构支撑功能安全开展 2&#xff09; 确保有足够的资源投入到功能安全开发中 3&#xff09; 有完整的功能安全培训 4&#xff09; 流程…

异常日志规范

目录 一、错误码 二、异常处理 三、日志规约 一、错误码 强制&#xff1a; 1、错误码的制订原则&#xff1a;快速溯源、沟通标准化。 1&#xff09;错误码必须能够快速知晓错误来源&#xff0c;可快速判断是谁的问题。 2&#xff09;错误码必须能够清晰地比对&#xff08;…

SOLID 面对象设计的五大基本原则

SOLID 原则的价值 原则核心价值解决的问题SRP职责分离&#xff0c;提高内聚性代码臃肿、牵一发而动全身OCP通过扩展而非修改实现变化频繁修改现有代码导致的风险LSP确保子类行为的一致性继承滥用导致的系统不稳定ISP定制化接口&#xff0c;避免依赖冗余接口过大导致的实现负担…

Python 装饰器详解

装饰器是 Python 中一种强大的语法特性&#xff0c;它允许在不修改原函数代码的情况下动态地扩展函数的功能。装饰器本质上是一个高阶函数&#xff0c;它接受一个函数作为参数并返回一个新的函数。 基本装饰器 1. 简单装饰器示例 def my_decorator(func):def wrapper():prin…

无损耗协议:PROFINET和EtherNet IP网关的高效安装指南

作为风力发电机组监控系统的重要组成部分&#xff0c;PROFINET和EtherNet/IP协议转换网关倍讯BX-606-EIP的安装至关重要。作为安装工,我们要确保网关安装的高效顺利,保证风力发电机组的稳定运行。 首先,我们需要仔细检查网关的硬件接口,确保所有连接线缆与设备端口相匹配。网关…

Axure元件动作四:设置选中

亲爱的小伙伴,在您浏览之前,烦请关注一下,在此深表感谢!如有帮助请订阅专栏! Axure产品经理精品视频课已登录CSDN可点击学习https://edu.csdn.net/course/detail/40420 课程主题:设置选中 主要内容:选中效果全面解析 应用场景:元件、元件组合需要被选中场景 案例展…

大模型为什么学新忘旧(大模型为什么会有灾难性遗忘)?

字数&#xff1a;2500字 一、前言&#xff1a;当学霸变成“金鱼” 假设你班上有个学霸&#xff0c;数学考满分&#xff0c;英语拿第一&#xff0c;物理称霸全校。某天&#xff0c;他突然宣布&#xff1a;“我要全面发展&#xff01;从今天起学打篮球&#xff01;” 一周后&am…

通过SMTP协议实现Linux邮件发送配置指南

一、环境准备与基础配置 1. SMTP服务开通&#xff08;以qq邮箱为例&#xff09; 登录qq邮箱网页端&#xff0c;进入「设置」-「POP3/SMTP/IMAP」 开启「SMTP服务」并获取16位授权码&#xff08;替代邮箱密码使用&#xff09; 记录关键参数&#xff1a; SMTP服务器地址&#…

react中安装依赖时的问题 【集合】

目录 依赖升级/更新 1、 npm install --save-dev 与 npm install 的区别 1. ‌安装位置&#xff08;依赖类型&#xff09;‌ 2. ‌package.json 中的区别‌ 3. ‌示例 4. ‌何时使用哪种方式‌ 2、npm install 和 yarn add 有什么不一样吗 ‌命令语法‌&#xff1a; …

Coze 实战教程 | 10 分钟打造你的AI 助手

> 文章中的 xxx 自行替换&#xff0c;文章被屏蔽了。 &#x1f4f1; 想让你的xxx具备 AI 对话能力&#xff1f;本篇将手把手教你&#xff0c;如何用 Coze 平台快速构建一个能与用户自然交流、自动回复提问的 xxx助手&#xff0c;零代码、超高效&#xff01; &#x1f4cc;…

【Spring Cloud Gateway】Nacos整合遇坑记:503 Service Unavailable

一、场景重现 最近在公司进行微服务架构升级&#xff0c;将原有的 Spring Cloud Hoxton 版本升级到最新的 2021.x 版本&#xff0c;同时使用 Nacos 作为服务注册中心和配置中心。在完成基础框架搭建后&#xff0c;我使用 Spring Cloud Gateway 作为API 网关&#xff0c;通过 N…

宝塔面板屏蔽垃圾搜索引擎蜘蛛和扫描工具的办法

首先进入宝塔面板&#xff0c;文件管理进入/www/server/nginx/conf目录&#xff0c;新建空白文件kill_bot.conf。然后将以下代码保存到当前文件中。 #禁止垃圾搜索引擎蜘蛛抓取if ($http_user_agent ~* "CheckMarkNetwork|Synapse|Nimbostratus-Bot|Dark|scraper|LMAO|Ha…