Spring MVC 深度解析:原理、源码剖析与实战

news2025/3/30 21:12:25

Spring MVC 深度解析:原理、源码剖析与实战

在 Spring 体系中,Spring MVC 作为 Web 层的核心框架,承担着请求处理、参数解析、视图渲染等关键任务。今天,我们将深入剖析 Spring MVC 的执行流程,结合 源码分析,并通过 代码实战 掌握其核心机制。


📌 1. 什么是 Spring MVC?

Spring MVC(Model-View-Controller)是一种基于 Servlet 的 Web 框架,遵循 MVC 设计模式,主要负责:

  • 请求分发(DispatcherServlet)
  • 控制器映射(HandlerMapping)
  • 参数解析(HandlerAdapter)
  • 视图渲染(ViewResolver)

📌 2. Spring MVC 执行流程解析

Spring MVC 处理请求的完整流程如下:

1️⃣ 客户端发送 HTTP 请求(如 GET /user/list
2️⃣ DispatcherServlet 拦截请求,作为 MVC 的中央调度器
3️⃣ HandlerMapping 查找对应的 Controller(基于 URL 映射)
4️⃣ HandlerAdapter 适配执行 Controller 业务逻辑
5️⃣ Controller 处理请求并返回 ModelAndView
6️⃣ ViewResolver 解析视图并渲染数据
7️⃣ 返回 HTML 响应给客户端

📌 Spring MVC 内部架构图

[Client] → [DispatcherServlet] → [HandlerMapping] → [Controller] → 
[ModelAndView] → [ViewResolver] → [Response]

📌 3. 源码解析:DispatcherServlet 如何工作?

🔥 DispatcherServlet 核心源码

Spring MVC 通过 DispatcherServlet 作为 前端控制器,其核心方法 doDispatch() 负责处理请求:

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
    // 1️⃣ 解析请求对应的 Handler(Controller)
    HandlerExecutionChain handler = getHandler(request);
    
    // 2️⃣ 选择合适的 HandlerAdapter 执行 Controller 方法
    HandlerAdapter ha = getHandlerAdapter(handler.getHandler());
    
    // 3️⃣ 调用 Controller 方法,获取 ModelAndView
    ModelAndView mv = ha.handle(request, response, handler.getHandler());

    // 4️⃣ 解析视图,并返回响应结果
    processDispatchResult(request, response, handler, mv);
}

核心流程

  • getHandler():查找 HandlerMapping 获取对应的 Controller
  • getHandlerAdapter():找到适配 Controller 方法的 HandlerAdapter
  • handle():执行 Controller 逻辑,返回 ModelAndView
  • processDispatchResult():调用 ViewResolver 解析视图,渲染响应

📌 4. 代码实战:手写 Spring MVC Controller

我们基于 Spring Boot 构建一个简单的 Spring MVC 应用,并实现 @RequestMapping 的 Controller。

📝 代码示例:Spring Boot MVC 控制器

import org.springframework.web.bind.annotation.*;

import java.util.*;

@RestController
@RequestMapping("/user")
public class UserController {
    
    // 模拟数据库存储用户
    private static List<String> users = new ArrayList<>(Arrays.asList("Alice", "Bob", "Charlie"));

    // 查询用户列表
    @GetMapping("/list")
    public List<String> listUsers() {
        return users;
    }

    // 添加用户
    @PostMapping("/add")
    public String addUser(@RequestParam String name) {
        users.add(name);
        return "用户 " + name + " 添加成功!";
    }
}

🛠 启动 Spring Boot

SpringBootApplication 类中运行:

@SpringBootApplication
public class SpringMvcApp {
    public static void main(String[] args) {
        SpringApplication.run(SpringMvcApp.class, args);
    }
}

测试 API

# 查询用户列表
curl http://localhost:8080/user/list

# 添加新用户
curl -X POST "http://localhost:8080/user/add?name=David"

📌 5. 深入分析 @RequestMapping 的执行流程

@RequestMapping 是 Spring MVC 用于 URL 映射 的核心注解,它的执行流程如下:

1️⃣ Spring MVC 启动时,扫描 @RequestMapping 标注的方法
2️⃣ HandlerMapping 解析 URL,并映射到 Controller 方法
3️⃣ HandlerAdapter 调用 Controller 方法,解析 @RequestParam 参数
4️⃣ 返回结果封装为 ResponseBody,通过 HttpMessageConverter 处理

📌 源码解析:Spring MVC 如何解析 @RequestMapping?

@RequestMapping("/user")
public class UserController {
    @GetMapping("/list")
    public List<String> listUsers() { return users; }
}

➡️ 核心逻辑
Spring MVC 在启动时,会通过 RequestMappingHandlerMapping 解析 @RequestMapping 并存入映射表:

protected void detectHandlerMethods(Object handler) {
    for (Method method : handler.getClass().getDeclaredMethods()) {
        if (method.isAnnotationPresent(RequestMapping.class)) {
            registerHandlerMethod(handler, method);
        }
    }
}

📌 6. Spring MVC 过滤器与拦截器

Spring MVC 提供 FilterInterceptor 机制,可用于 用户认证、日志记录、跨域请求 等场景。

类型作用实现方式
Filter过滤 HTTP 请求javax.servlet.Filter
Interceptor拦截 ControllerHandlerInterceptor

📝 代码示例:拦截器实现日志记录

@Component
public class LogInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
        System.out.println("请求路径:" + request.getRequestURI());
        return true;
    }
}

注册拦截器

@Configuration
public class WebConfig implements WebMvcConfigurer {
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new LogInterceptor());
    }
}

🔹 作用:每次请求前打印日志,方便调试。


📌 7. 总结

功能Spring 机制源码解析代码实战
请求分发DispatcherServletdoDispatch() 方法@RequestMapping
Controller 处理HandlerMappinggetHandler()@RestController
参数解析HandlerAdapterhandle() 方法@RequestParam
视图渲染ViewResolverprocessDispatchResult()return ModelAndView

今日收获

  • 掌握 Spring MVC 执行流程
  • 深入理解 DispatcherServlet 处理逻辑
  • 解析 @RequestMapping 的工作机制
  • 实战 Spring Boot MVC 开发
  • 实现 日志拦截器

🚀 明日预告:Spring 事务管理(事务传播、@Transactional 源码分析)

🔗 学习资料
📖 Spring 官方文档 - MVC


💬 Spring MVC 你有什么疑问?欢迎留言交流! 🚀

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

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

相关文章

【NLP 44、实践 ⑪ 用Bert模型结构实现自回归语言模型的训练】

目录 数据文件 一、模型定义 1.模型初始化 代码运行流程 2.前向传播&#xff0c;计算损失 ⭐ 代码运行流程 二、加载语料 代码运行流程 三、 随机生成样本 代码运行流程 四、建立模型 五、采样策略选择 代码运行流程 六、模型效果测试 代码运行流程 七、模型训练 代码运行流程 …

微信小程序如何接入直播功能

一、小程序直播开通背景 1.政府资质要求 政府的要求&#xff0c;小程序开通直播需要注册主体具备互联网直播的资质&#xff0c;普通企业需要《信息网络传播视听节目许可证》&#xff0c;表演性质的直播需要《网络文化经营许可证》&#xff0c;政府主体需要《社会信用代码》及…

基于Spring Boot的停车场管理系统的设计与实现(LW+源码+讲解)

专注于大学生项目实战开发,讲解,毕业答疑辅导&#xff0c;欢迎高校老师/同行前辈交流合作✌。 技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、小程序、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容&#xff1a;…

慧通测控汽车智能座舱测试技术

一、引言 随着科技的飞速发展&#xff0c;汽车正从单纯的交通工具向智能化移动空间转变。智能座舱作为这一转变的核心体现&#xff0c;融合了多种先进技术&#xff0c;为用户带来前所未有的驾驶体验。从简单的信息娱乐系统到高度集成的人机交互、智能驾驶辅助以及车辆状态监测…

kettle插件-rabbitmq插件

场景&#xff1a;kettle本身可以直接链接rabbitmq&#xff0c;但是需要配置rabbitmq开启mqtt协议&#xff0c;本次讲解下自定义开发组件RabbitMQ consumer&#xff0c;无需开启mqtt协议即可使用。 1、docker 安装rabbitmq 1&#xff09;下载镜像 docker pull rabbitmq 2&…

为Windows10的WSL Ubuntu启动sshd服务并使用Trae远程连接

Windows10的WSL Ubuntu&#xff0c;使用起来非常方便&#xff0c;但是美中不足的是&#xff0c;无法从Windows主机ssh到Ubuntu 。 解决的方法是在Ubuntu安装sshd服务 Ubuntu安装sshd服务 执行命令 sudo apt install openssh-server 安装好后&#xff0c;先本地测试&#x…

【C#.NET】VS2022创建Web API项目

C# Web API 是一种基于 .NET 平台&#xff08;包括但不限于.NET Framework 和 .NET Core&#xff09;构建 HTTP 服务的框架&#xff0c;用于创建 RESTful Web 服务。REST&#xff08;Representational State Transfer&#xff09;是一种软件架构风格&#xff0c;它利用HTTP协议…

体育直播系统趣猜功能开发技术实现方案

功能概述 趣猜功能是“东莞梦幻网络科技”体育直播系统源码中的互动功能&#xff0c;主播可以发起竞猜题目&#xff0c;观众使用虚拟货币进行投注&#xff0c;增加直播间的互动性和趣味性。所有货币均为虚拟货币&#xff0c;通过系统活动获取&#xff0c;不可充值提现。 数据…

33.[前端开发-JavaScript基础]Day10-常见事件-鼠标事件-键盘事件-定时器-案例

1 window定时器 window定时器方法 setTimeout的使用 setInterval的使用 2 轮播消息提示 案例实战一 – 轮播消息提示 3 关闭隐藏消息 案例实战二 – 关闭隐藏消息 4 侧边栏展示 案例实战三 – 侧边栏展示 5 tab切换实现 案例实战四 – 登录框&#xff08;作业&#xff09;…

C# 多标签浏览器 谷歌内核Csharp

采用框架 &#xff1a;FBrowserCEF3lib 视频演示&#xff1a;点我直达 成品下载&#xff1a; https://wwms.lanzouo.com/iYOd42rl8vje

如何从0设计开发一款JS-SDK

一、前言 前端SDK是什么&#xff1f;前端SDK是为了帮助前端实现特定需求&#xff0c;而向开发者暴露的一些JS-API的集合&#xff0c;规范的SDK包括若干API实现、说明文档等 前端SDK其实很常见了&#xff0c;比如&#xff1a; UI组件库&#xff1a;通过封装一系列组件&#xff…

linux实现rsync+sersync实时数据备份

1.概述 rsync(Remote Sync) 是一个Unix/linux系统下的文件同步和传输工具 2.端口和运行模式 tcp/873 采用C/S模式&#xff08;客户端/服务器模式&#xff09; 3.特点 可以镜像保存整个目录和文件第一次全量备份(备份全部的文件),之后是增量备份(只备份变化的文件) 4. 数…

【计算机网络】计算机网络协议、接口与服务全面解析——结合生活化案例与图文详解

协议、接口与服务 导读一、协议1.1 定义1.2 组成 二、接口三、服务3.1 定义3.2 服务与协议的区别3.3 分类3.3.1 面向连接服务于无连接服务3.3.2 可靠服务和不可靠服务3.3.3 有应答服务和无应答服务 结语 导读 大家好&#xff0c;很高兴又和大家见面啦&#xff01;&#xff01;…

51c自动驾驶~合集26

我自己的原文哦~ https://blog.51cto.com/whaosoft/11968755 #大模型/Sora/世界模型之间是什么关系 1 什么是大模型 人工智能大模型&#xff08;Artificial Intelligence Large Model&#xff0c;简称AI大模型&#xff09;是指具有庞大的参数规模和复杂程度的机器学习模…

【NUUO 摄像头】(弱口令登录漏洞)

漏洞简介&#xff1a;NUUO 是NUUO公司的一款小型网络硬盘录像机设备。 NUUO NVRMini2 3.0.8及之前版本中存在后门调试文件。远程攻击者可通过向后门文件handle_site_config.php发送特定的请求利用该漏洞执行任意命令。 1.Fofa搜索语句&#xff1a; 在Fofa网站&#xff0c;搜索&…

【设计模式】抽象工厂模式(含与工厂方法模式的对比)

本期我们来学习一下设计模式之抽象工厂模式&#xff0c;在软件开发中&#xff0c;工厂模式 和 抽象工厂模式 都用于创建对象&#xff0c;但它们的应用场景和实现方式有所不同。本文将基于 C 代码&#xff0c;分析抽象工厂模式的实现&#xff0c;并对比其与工厂方法模式的区别。…

IDEA转战Trae AI IED配置

Trae Ai 的前身是vscode IDEA转战Trae AI IED配置 1.安装java相关的插件 2、安装spring相关的插件 3.配置maven环境 打开 Trae AI IDE -> 首选项 -> 设置 -> Editor 设置 ⚠️配置方式有两种 setting.json文件中直接编辑&#xff08;推荐&#xff09;界面设置 方案…

再学:区块链基础与合约初探 EVM与GAS机制

目录 1.区块链是什么 2.remix ​3.账户​ ​4.以太坊三种交易​ 5.EVM 6.以太坊客户端节点 ​7.Gas费用 8.区块链浏览器 1.区块链是什么 只需要检验根节点 Merkel根是否有更改&#xff0c;就不用检查每个交易是否有更改。方便很多。 2.remix 3.账户 如果交易失败的话&…

Nextjs15 - middleware的使用

nextjs 官方文档&#xff08;current branch 对应如下文档&#xff09; Middlewarepath-to-regexp 本专栏内容均可在Github&#xff1a;test_05/Middleware 找到 一、middleware 基本使用 中间件允许您在请求完成之前运行代码。然后&#xff0c;根据传入的请求&#xff0c;您…

边缘计算 vs. 云计算,谁才是工业物联网的未来?

前言 在物联网&#xff08;IoT&#xff09;飞速发展的今天&#xff0c;边缘计算正在彻底改变数据的处理、存储和分析方式。传统的IoT设备数据通常需要发送到云端进行处理&#xff0c;但随着设备数量的激增&#xff0c;这种模式在延迟、带宽和安全性方面暴露出诸多局限。边缘计…