文章目录

    • 一、双容器架构:MVC容器与根容器的关系
    • 二、启动全流程解析
      • 1. 启动流程全景图
      • 2. 初始化根容器(Root WebApplicationContext)
        • 2.1 Tomcat 中启动入口源码解析
        • 2.2 Spring 根上下文启动源码解析
      • 3. 初始化 MVC 容器(DispatcherServlet 的 WebApplicationContext)
        • 3.1 Tomcat 中启动入口源码解析
        • 3.2 Spring MVC上下文启动源码解析
          • 1. ★MVC容器初始化入口:HttpServletBean
          • 2. 创建 MVC 容器:FrameworkServlet
          • 3. 核心逻辑:initWebApplicationContext()
          • 4. 创建MVC子容器:createWebApplicationContext()
          • 5. 配置并刷新容器:configureAndRefreshWebApplicationContext()
          • 6. 初始化 MVC 组件:DispatcherServlet 的 onRefresh()
        • 3.3 核心启动流程
      • 4. 关键设计解析
        • 1. 父子容器设计的优势
        • 2. 设计意义与价值
    • 三、调试技巧
    • 四、总结
    • 扩展
      • DispatcherServlet Diagram
      • Tomcat 中的完整调用栈
      • Servlet 3.0+ 无配置启动(Java Config)

在Java Web开发中,理解Spring MVC如何与Tomcat等Web容器协同工作是掌握企业级应用开发的关键。本文将深入解析Spring MVC容器在Web容器中的启动过程,揭示父子容器协作的奥秘。

在上一篇中详细介绍了 Spring IOC容器在web容器中的启动过程,这篇进一步了解下Spring MVC容器(即 Web 应用上下文)是如何在web容器(如 Tomcat、Jetty)中启动并生效的。在 Spring MVC 中,MVC 容器(即 DispatcherServletWeb 应用上下文)的初始化过程是一个精密的协作机制。以下是详细的启动流程和关键代码调用:

一、双容器架构:MVC容器与根容器的关系

Spring MVC采用父子容器设计,实现业务层与Web层的关注点分离:
在这里插入图片描述

  • 根容器:由ContextLoaderListener创建,管理业务层和数据层Bean
  • MVC容器:由DispatcherServlet创建,管理Web层组件
  • 依赖规则:子容器可访问父容器的Bean,反之则不行

二、启动全流程解析

1. 启动流程全景图

在这里插入图片描述

Web 容器启动

  • Web 容器(如 Tomcat)启动时,会加载 web.xml(或 Servlet 3.0+ 的注解配置)。
  • 容器根据配置初始化 ServletContext(全局上下文),作为整个 Web 应用的共享空间。
  • ServletContext 基础上展开Spring Web 容器的一系列启动初始化

2. 初始化根容器(Root WebApplicationContext)

2.1 Tomcat 中启动入口源码解析

源码入口StandardContext.startInternal()

在这里插入图片描述
在这里插入图片描述

核心点:上面 Tomcat 源码中 listener.contextInitialized(event) 方法会执行到 Spring ContextLoaderListener.contextInitialized()方法, 从而初始化Spring Web 根上下文(IOC容器),建立起在Web环境中Spring IOC容器。

2.2 Spring 根上下文启动源码解析

源码入口ContextLoaderListener.contextInitialized()

[提示]:详细解析过程可查阅: Spring IOC容器在web容器中的启动过程

3. 初始化 MVC 容器(DispatcherServlet 的 WebApplicationContext)

Tomcat 等 Web 容器(Servlet 容器)启动时调用 Servletinit() 方法是一个由 Java Servlet 规范定义的标准过程,其细节如下:

3.1 Tomcat 中启动入口源码解析

源码入口StandardContext.startInternal()

在这里插入图片描述
Servlet 加载机制: StandardContext.loadOnStartup()
在这里插入图片描述
核心代码:StandardWrapper.loadServlet()
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

核心点:上面的 Tomcat 源码中GenericServlet.init()方法实际会调用到 Spring DispatcherServlet.load()方法(DispatcherServlet的继承链:DispatcherServlet → FrameworkServlet → HttpServletBean→ HttpServlet→ GenericServlet),从而初始化Spring MVC 子上下文(Web IOC容器),建立起在Web环境中Spring MVC架构来接收处理 HTTP 请求。

疑问点:为什么调用 servlet.init()?

  • Servlet 规范要求,所有 Servlet 必须实现 javax.servlet.Servlet 接口;Java Servlet 规范(JSR 369)明确定义:
  • “After the servlet object is instantiated, the container must initialize the servlet before it can handle requests. The container initializes the servlet by calling the init(ServletConfig) method.”

  • load-on-startup 控制:在 web.xml 中配置的 <load-on-startup> 决定初始化时机;
    • 延迟加载(默认行为):在 Tomcat 容器启动时,默认情况下 不会立即初始化 ServletServlet 的初始化通常是延迟的(lazy loading),即在第一次接收到与该 Servlet 相关的请求时才会进行初始化。这种行为是由 Servlet 规范定义的,目的是为了节省资源。
    • 启动时加载(eager loading):如果在 web.xml 中为 Servlet 配置了 <load-on-startup> 元素,Tomcat 会在容器启动时初始化该 Servlet。其值为一个整数,表示加载顺序。0或正值,值越小,优先级越高。负值或未指定,首次请求时初始化。
3.2 Spring MVC上下文启动源码解析

源码入口DispatcherServlet.init()

  • 配置web.xml:在这里插入图片描述

由于继承关系,实际初始化入口类为DispatcherServlet的父类HttpServletBean,源码位置:org.springframework.web.servlet.HttpServletBean

1. ★MVC容器初始化入口:HttpServletBean

在这里插入图片描述

2. 创建 MVC 容器:FrameworkServlet

在这里插入图片描述

3. 核心逻辑:initWebApplicationContext()

在这里插入图片描述

4. 创建MVC子容器:createWebApplicationContext()

在这里插入图片描述
在这里插入图片描述

5. 配置并刷新容器:configureAndRefreshWebApplicationContext()

在这里插入图片描述

6. 初始化 MVC 组件:DispatcherServlet 的 onRefresh()

在这里插入图片描述

3.3 核心启动流程
  • DispatcherServlet 初始化时调用 init() 方法。
  • 创建 子应用上下文(专用于 Web 层的容器),自动将根上下文设置为父容器。
  • 加载 contextConfigLocation 指定的 MVC 配置(如 Controller、视图解析器等)。
  • 刷新子上下文(refresh() 方法),初始化所有 MVC 相关的 组件Bean。

通过此流程,Spring MVC 实现了 Web 层组件的精确控制,同时通过父子容器隔离了业务层与 Web 层的 Bean 管理。

4. 关键设计解析

1. 父子容器设计的优势
  • 关注点分离:业务层与Web层解耦
  • 资源隔离:避免Controller污染业务层
  • 灵活配置:不同容器可独立配置
  • 依赖可控:子容器可访问父容器,反之不行
  • 独立刷新:Web层重启不影响业务层
2. 设计意义与价值
  1. 生命周期管理
    • 容器完全控制 Servlet 的创建 → 初始化 → 服务 → 销毁
    • 保证资源有序初始化和释放
  2. 依赖解耦
    • Servlet 无需知道容器实现细节
    • 通过标准接口 ServletConfig 获取配置
  3. 资源预加载
    • load-on-startup 避免首次请求延迟
    • 特别适合 Spring MVC 这类重量级前端控制器
  4. 扩展性
    • Spring 通过重写 init() 插入自定义初始化逻辑
    • 实现父子容器、组件初始化等高级特性

三、调试技巧

关键断点位置:

  • FrameworkServlet.initWebApplicationContext()
  • AbstractApplicationContext.refresh()
  • DispatcherServlet.initStrategies()
  • RequestMappingHandlerMapping.afterPropertiesSet()

四、总结

Spring MVC在Web容器中的启动是一个精密的协作过程:整个启动过程Servlet 规范 驱动(监听器、Servlet 生命周期),Spring 在此基础上扩展上下文层次。

  • Tomcat 通过load-on-startup机制触发Servlet初始化
  • ContextLoaderListener 创建根容器管理业务Bean
  • DispatcherServlet 创建子容器管理Web组件。
  • 父子容器 通过setParent()建立层级关系。
  • 父子容器refresh() 方法触发完整的Bean初始化流程

通过这种分层设计,Spring 实现了关注点分离(业务层 vs Web 层),同时确保依赖注入的正确性。 通过这套机制,Tomcat
等容器保证了 Spring MVC 这类框架能在正确的时间点初始化自己的核心组件,同时遵循 Java EE 标准规范。


End!

扩展

DispatcherServlet Diagram

在这里插入图片描述

Tomcat 中的完整调用栈

// Tomcat 启动入口
Bootstrap.main()Catalina.load()StandardServer.start()StandardService.start()StandardEngine.start()StandardHost.start()StandardContext.start()StandardContext.startInternal()StandardContext.fireLifecycleEvent()  // 触发监听器(初始化Spring Web根容器)StandardContext.loadOnStartup()       // 关键:启动时加载ServletStandardWrapper.loadServlet()DispatcherServlet.init(ServletConfig)  // Spring MVC子容器入口HttpServletBean.init()FrameworkServlet.initServletBean()initWebApplicationContext()  // 初始化MVC容器

Servlet 3.0+ 无配置启动(Java Config)

通过实现 WebApplicationInitializer 接口替代 web.xml

public class MyWebAppInitializer implements WebApplicationInitializer {@Overridepublic void onStartup(ServletContext servletContext) {// 1. 创建根容器AnnotationConfigWebApplicationContext rootContext = new AnnotationConfigWebApplicationContext();rootContext.register(RootConfig.class);servletContext.addListener(new ContextLoaderListener(rootContext));// 2. 创建 MVC 容器(子容器)AnnotationConfigWebApplicationContext mvcContext = new AnnotationConfigWebApplicationContext();mvcContext.register(WebConfig.class);// 3. 注册 DispatcherServletDispatcherServlet servlet = new DispatcherServlet(mvcContext);ServletRegistration.Dynamic registration = servletContext.addServlet("dispatcher", servlet);registration.addMapping("/");registration.setLoadOnStartup(1);}
}

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

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

相关文章

【iOS】编译和链接、动静态库及dyld的简单学习

文章目录编译和链接1️⃣核心结论&#xff1a;一句话区分2️⃣编译过程&#xff1a;从源代码到目标文件&#xff08;.o&#xff09;2.1 预处理&#xff08;Preprocessing&#xff09;&#xff1a;“替换变量复制粘贴”2.2 编译&#xff08;Compilation&#xff09;&#xff1a;…

金山办公WPS项目产品总监陈智新受邀为第十四届中国PMO大会演讲嘉宾

全国PMO专业人士年度盛会珠海金山办公软件有限公司WPS项目产品总监 陈智新先生 受邀为“PMO评论”主办的2025第十四届中国PMO大会演讲嘉宾&#xff0c;演讲议题为&#xff1a;中小团队PMO的成长之路&#xff0c;敬请关注&#xff01;议题简要&#xff1a;在竞争激烈、需求多变的…

web安全 | docker复杂环境下的内网打点

本文作者&#xff1a;Track-syst1m一.前言本文涉及的相关漏洞均已修复、本文中技术和方法仅用于教育目的&#xff1b;文中讨论的所有案例和技术均旨在帮助读者更好地理解相关安全问题&#xff0c;并采取适当的防护措施来保护自身系统免受攻击。二.大概流程1. 外网打点• 漏洞利…

iTwin 几何属性获取

面积体积半径获取几何属性&#xff0c;如面积&#xff0c;体积&#xff0c;半径&#xff0c;可以使用getMassProperties这个接口async onGetMassProperty(){const vp IModelApp.viewManager.selectedView;const iModel vp?.iModel;if (!iModel) return;console.log("iM…

OpenLayers 快速入门(九)Extent 介绍

看过的知识不等于学会。唯有用心总结、系统记录&#xff0c;并通过温故知新反复实践&#xff0c;才能真正掌握一二 作为一名摸爬滚打三年的前端开发&#xff0c;开源社区给了我饭碗&#xff0c;我也将所学的知识体系回馈给大家&#xff0c;助你少走弯路&#xff01; OpenLayers…

LeetCode 121. 买卖股票的最佳时机 LeetCode 122. 买卖股票的最佳时机II LeetCode 123.买卖股票的最佳时机III

LeetCode 121. 买卖股票的最佳时机尝试一&#xff1a;暴力解决方法常用两个指针去遍历prices数组&#xff0c;dp[i]用于记录在第i天所获得的最大利润。时间复杂度是O(N^2)&#xff0c;超出时间限制。Codeclass Solution(object):def maxProfit(self, prices):"""…

【LeNet网络架构】——深度学习.卷积神经网络

目录 1 MLP 2 LeNet简介 3 Minst数据集 3.1 MINST数据集简介 3.2 MNIST数据集的预处理 4 LeNet手写数字识别 LeNet由Yann Lecun 提出&#xff0c;是一种经典的卷积神经网络&#xff0c;是现代卷积神经网络的起源之一。Yann将该网络用于邮局的邮政的邮政编码识别&#xff…

Python笔记完整版

常用pip源 &#xff08;1&#xff09;阿里云 http://mirrors.aliyun.com/pypi/simple/&#xff08;2&#xff09;豆瓣 http://pypi.douban.com/simple/&#xff08;3&#xff09;清华大学 https://pypi.tuna.tsinghua.edu.cn/simple/&#xff08;4&#xff09;中国科学技术大学…

2025 鸿蒙创新赛又来了,万少教你如何强势切入 HarmonyOS AI特性

2025 鸿蒙创新赛又来了&#xff0c;万少教你如何强势切入 前言 ​ 2025 华为HarmonyOS 创新赛又来了&#xff0c;创新赛是鸿蒙生态最大规模开发者官方赛事&#xff0c;最高获百万激励。 参赛资格 面向所有开发者开放以队伍的形式来参加&#xff0c;可以一个人报名一个队伍&a…

【智能模型系列】Unity通过访问Ollama调用DeepSeek模型进行本地部署

【智能模型系列】Unity通过访问Ollama调用DeepSeek模型进行本地部署 目录 一、前言 二、环境准备 三、核心代码解析 1、参数配置 2. CallDeepSeek.cs - API交互控制器 3、 MainPanel.cs - 用户界面控制器 四、源码 一、前言 在本教程中,我将分享如何在Unity中集成本地…

什么是5G-A三防平板?有什么特点?哪些领域能用到?

在工业自动化与数字化转型浪潮中&#xff0c;三防平板电脑已成为“危、急、特”场景的核心工具。这类设备不仅具备坚固耐用的物理防护特性&#xff0c;更融合了先进的通信技术与智能处理能力。而随着5G技术向5G-A阶段演进&#xff0c;新一代三防平板正为行业应用注入全新动能。…

Flink实时流量统计:基于窗口函数与Redis Sink的每小时PV监控系统(学习记录)

题目&#xff1a;利用flink统计网站浏览量&#xff0c;并写入redis。利用窗口函数以及算子实现每小时PV&#xff08;网站的页面浏览量&#xff09;统计&#xff0c;对统计后结果数据格式进行设计&#xff0c;存储至Redis中&#xff08;利用sink将处理后结果数据输出到redis数据…

使用Imgui和SDL2做的一个弹球小游戏-Bounze

使用Imgui和SDL2做的一个弹球小游戏-Bounze 油管上面TheCherno博主分享的一个视频FIRST GAME in C! Did He Do a Good Job? // Code Review (C/SDL2)里面分享了一个Github项目&#xff1a; https://github.com/staticaron/Bounze 使用了Imgui和SDL2&#xff0c;并且可以设置音…

SQL 中 CASE WHEN 及 SELECT CASE WHEN 的用法

SQL 中 CASE WHEN 及 SELECT CASE WHEN 的用法 CASE WHEN 是 SQL 中非常实用的条件表达式&#xff0c;它允许你在查询中实现条件逻辑。以下是详细的用法说明&#xff1a; 1. 基本语法结构 CASE WHEN condition1 THEN result1WHEN condition2 THEN result2...ELSE default_resul…

CentOS 7 Linux 基础知识点汇总

&#x1f427; CentOS 7 Linux 基础知识点汇总为方便初学者快速掌握 CentOS 7 系统的核心操作&#xff0c;本文档整理了常用系统命令、快捷键、目录结构及文件后缀名等基础内容&#xff0c;适合入门参考。 一、常见系统命令 &#x1f50d; 命令行提示符说明 终端中的提示符包含…

突发限制下的破局之路:国产之光 Lynx 重构 AI 开发安全壁垒

继 Pro 套餐 “明升暗降” 争议后&#xff0c;Cursor 本周再掀波澜 —— 包括 Claude 系列、GPT-4 在内的主流模型一夜之间对中国用户全面封禁。开发者社群瞬间沸腾&#xff0c;“付费却用不了”“项目数据导不出” 的焦虑刷屏&#xff0c;境外工具的政策波动再次给行业敲响警钟…

渗透测试实战 | docker复杂环境下的内网打点

本文作者&#xff1a;Track-syst1m一.前言本文涉及的相关漏洞均已修复、本文中技术和方法仅用于教育目的&#xff1b;文中讨论的所有案例和技术均旨在帮助读者更好地理解相关安全问题&#xff0c;并采取适当的防护措施来保护自身系统免受攻击。二.大概流程1. 外网打点漏洞利用•…

阿里云服务器 CentOS 7 安装 MySQL 8.4 超详细指南

阿里云服务器 CentOS 7 安装 MySQL 8.4 超详细指南 一、准备工作 系统要求&#xff1a; CentOS 7.9 64位2 核&#xff08;vCPU&#xff09;2 GiBroot 用户权限 服务器连接工具&#xff1a; FinalShell 下载安装包&#xff1a; 访问 MySQL 官网选择版本&#xff1a;MySQL 8.4.0…

解决 Electron 中 window.open 打开新窗口的各种“坑”

嘿&#xff0c;各位开发者们&#xff01;今天我们要聊聊在使用 Electron 时遇到的一个经典问题&#xff1a;如何正确地使用 window.open 来打开新窗口&#xff1f; 这听起来似乎很简单&#xff0c;但实际上却充满了各种“惊喜”&#xff08;或者说“惊吓”&#xff09;。别担心…

朝歌智慧盘古信息:以IMS MOM V6重构国产化智能终端新生态

随着5G、云计算、AI、大数据等技术深度渗透&#xff0c;智能终端行业正迎来场景化创新的爆发期。面对市场需求升级与技术迭代压力&#xff0c;国产化智能终端领域领军企业——广东朝歌智慧互联科技有限公司&#xff08;以下简称“朝歌智慧”&#xff09;&#xff0c;基于集团“…