springmvc源码流程解析(一)
Springmvc 是基于servlet 规范来完成的一个请求响应模块,也是spring 中比较大的一个 模块,现在基本上都是零xml 配置了,采用的是约定大于配置的方式,所以我们的springmvc 也是采用这种零xml 配置的方式。
要完成这种过程,要解决两个问题:1、取代web.xml 配置 ,2、取代springmvc.xml 配置。 取代web.xml 配置在servlet 中有一个规范,就是当servlet 容器启动的时候会根据SPI 规范加载 META-INF/services 文件夹下面的javax.servlet.ServletContainerInitializer 文件,该文件下面的 类会实现javax.servlet.ServletContainerInitializer 接口。
SPI 也被称为服务接口扩展,(Service Provider Interface) 直译服务提供商接口, 不要被这个名字唬到了, 其实很好理解的一个东西: 就是根据Servlet厂商(服务提供商)提供要求的一个接口, 在固定的目录 (META-INF/services)放上以接口全类名 为命名的文件, 文件中放入接口的实现的 全类名,该类由我们自己实现,按照这种约定的方式(即SPI规范),服务提供商会 调用文件中实现类的方法, 从而完成扩展。
该类SpringServletContainerInitializer在启动的时候会被servlet 容器(例如tomcat)实例化,然后调用onStartup 方法,并且servlet 容器会收集实现了@HandlesTypes 注解里面的接口的类(WebApplicationInitializer),并且做为入参传入到onStartup 方法中,我们拿到set 容器中的类就可以反射调用接口里面的方法了,这是servlet 规范,该规范就能保证servlet 容器在启动的时候就会完成这些操作。Springmvc 就借助这一点完成了取代web.xml 的工作。
我们定义了一个类MyWebAppInitializer继承自AbstractAnnotationConfigDispatcherServletInitializer,最上层的接口其实就是WebApplicationInitializer,这个接口有两个重要实现getRootConfigClasses将RootConfig引入了进来,这个RootConfig就是父容器的配置类,相当于之前的spring.xml;另一个方法getServletConfigClasses引入进来了WebAppConfig,这个是是子容器的配置类,相当于之前的spring-mvc.xml。
父容器会对com.dsk下的包进行扫描,但是排除了类上注解为@RestController和@Controller的类。因为controller要交给springmvc自身的子容器去管理,父容器会管理一些@Service相关的类。
子容器会对com.dsk下的包进行扫描,但是只扫描了类上注解为@RestContrller和@Controller的类
相关的类介绍完之后,看下核心的onStartup方法,我们定义的MyWebAppInitializer与其父类的继承关系分别是MyWebAppInitializer-》AbstractAnnotationConfigDispatcherServletInitializer-》AbstractDispatcherServletInitializer-》AbstractContextLoaderInitializer-》WebApplicationInitializer
第一步继续会调用父类的onStartUp方法
第二步注册dispatcherServlet
我们再往下一层AbstractContextLoaderInitializer父类的onStartup方法,可以看到这一步就是注册监听器ContextLoaderListener,注册进servletContext,这个监听器的作用其实就是当servlet容器(例如tomcat)启动成功后,会调用监听器的contextInitialized方法。
这一步相当于在web.xml中配置了ContextLoaderListener,如下
registerContextLoaderListener方法中不只是注册了listener,还创建了父容器rootAppContext
创建父容器的方法createRootApplicationContext中,其实就是熟悉的AnnotationConfigWebApplicationContext,他将父容器也就是RootConfig注册进了BeanDefinitionRegistry中
总结registerContextLoaderListener方法做了两件事情
1、创建父容器,也就是spring的AnnotationConfigWebApplicationContext
2、将ContextLoaderListener加入到servletContet中,注意此时ContextLoaderListener持有父容器对象rootContext方法,目的是当servlet容器启动成功后,会通知到监听器,此时监听器会调用父容器的refresh方法,将类注入到spring容器中。
那么父类AbstractContextLoaderInitializer的onStartUp方法就看完了,回到AbstractDispatcherServletInitializer类
这一步就是将dispatcherServlet注册进servletContext中,并把上下文对象设置到了
dispatcherServlet 对象中,相当于web.xml的如下配置
这个方法registerDispacherServlet不止是将dispatcherServlet加入到了servletContext,同时也创建了子容器,可以看到这个子容器也是AnnotationConfigWebApplicationContext,只不过此时注册进来的类是WebAppConfig,子类会将带有@Controller的类扫描进来,注入到子容器中。
那么onStartUp方法就执行完了,做个总结:
一、registerContextLoaderListener(servletContext);
做了两件事情1、创建父容器,且listener中持有父容器对象;2、将ContextLoaderListener加入到ServletContext
二、registerDispatcherServlet(servletContext);
做了两件事情1、创建子容器,且dispatcherServlet中持有子容器对象;2、将DispatcherServlet加入到ServletContext;
ContextLoaderListener启动:
执行完onStartUp方法后,此时父容器和子容器还没有启动,因为并没有执行到他们的refresh方法,我们知道当servlet容器(tomcat启动成功后)会调用监听器ContextLoaderListener.contextInitialized方法,这里会启动spring容器,把spring上下文对象放入到了servletContext中
DispatcherServlet 的启动:
了解DispatcherServlet的生命周期的会知道,DispatcherServlet启动过程中会执行init()方法,在这里会进行springmvc子容器的启动
这里会从servletContext中获取父容器对象,并将子容器webApplicationContext的parent赋值为rootContext,然后执行子容器的refresh()方法。
在调用子容器的刷新方法前,这里会注册一个监听器(该监听会初始化springmvc所需信息),
ContextRefreshedEvent可以看到该监听器监听的是容器refreshed事件, 会在子容器的refresh方法中的finishRefresh中发布,这是spring的监听器模式。
子容器的refresh()方法最后一步,会发布事件触发监听器的执行
这里面的每一个方法不用太细看, 就是给SpringMVC准备初始化的数据, 为后续SpringMVC处理请求做准备基本都是从容器中拿到已经配置的Bean(RequestMappingHandlerMapping、
RequestMappingHandlerAdapter、HandlerExceptionResolver )放到dispatcherServlet中做准备。但是这些Bean又是从哪来的呢?? 回到我们的WebAppConfig使用的@EnableWebMvc,也就是这个注解取代了springmvc.xml
1. 导入了DelegatingWebMvcConfiguration@Import(DelegatingWebMvcConfiguration.class)
2. DelegatingWebMvcConfiguration的父类就配置了这些Bean
3. SpringBoot也是用的这种方式;
总结:
1. Tomcat在启动时会通过SPI注册 ContextLoaderListener和DispatcherServlet对象
a. 同时创建父子容器
i. 分别创建在ContextLoaderListener初始化时创建父容器设置配置类
ii. 在DispatcherServlet初始化时创建子容器设置配置类
2. Tomcat在启动时执行ContextLoaderListener和DispatcherServlet对象的初始化方
法, 执行容器refresh进行加载
3. 在子容器加载时 创建SpringMVC所需的Bean和预准备的数据:(通过配置类
@EnableWebMvc配置(DelegatingWebMvcConfiguration)——可实现WebMvcConfigurer进行定制扩展)
a. RequestMappingHandlerMapping,它会处理@RequestMapping 注解
b. RequestMappingHandlerAdapter,则是处理请求的适配器,确定调用哪个类的哪个方法,并且构造方法参数,返回值。
c. HandlerExceptionResolver 错误视图解析器
d. addDefaultHttpMessageConverters 添加默认的消息转换器(解析json、解析xml)等
4. 子容器需要注入父容器的Bean时(比如Controller中需要@Autowired例如Service的Bean); 会先从子容器中找,没找到会去父容器中找。
一、Spring和SpringMVC为什么需要父子容器?不要不行吗?就实现层面来说不用子父容器也可以完成所需功能(参考:SpringBoot就没用子父容器)?
1. 所以父子容器的主要作用应该是早期Spring为了划分框架边界。有点单一职责的味道service、dao层我们一般使用spring框架来管理、controller层交给springmvc管理
2. 规范整体架构 使 父容器service无法访问子容器controller、子容器controller可以访问父容器 service
3. 方便子容器的切换。如果现在我们想把web层从spring mvc替换成struts,那么只需要将springmvc.xml替换成Struts的配置文件struts.xml即可,而spring.xml不需要改变。
4. 为了节省重复bean创建
二、是否可以把所有Bean都通过Spring容器来管理?(Spring的applicationContext.xml中配置全局扫描)?
不可以,这样会导致我们请求接口的时候产生404。 如果所有的Bean都交给父容器,SpringMVC在初始化HandlerMethods的时候(initHandlerMethods)无法根据Controller的handler方法注册HandlerMethod,并没有去查找父容器的bean;也就无法根据请求URI 获取到 HandlerMethod来进行匹配.
三、是否可以把我们所需的Bean都放入Spring-mvc子容器里面来管理(springmvc的springservlet.xml中配置全局扫描)?
可以 , 因为父容器的体现无非是为了获取子容器不包含的bean, 如果全部包含在子容器完全用
不到父容器了, 所以是可以全部放在springmvc子容器来管理的。虽然可以这么做不过一般应该是不推荐这么去做的,一般人也不会这么干的。如果你的项目里有用到事物、或者aop记得也需要把这部分配置需要放到Spring-mvc子容器的配置文件来,不然一部分内容在子容器和一部分内容在父容器,可能就会导致你的事务或者AOP不生效。 所以如果aop或事务如果不生效也有可能是通过父容器中的类(spring)去增强子容器的类(Springmvc),也就无法增强。
下个章节我们解析springmvc的请求源码流程。
相关文章:
springmvc源码流程解析(一)
Springmvc 是基于servlet 规范来完成的一个请求响应模块,也是spring 中比较大的一个 模块,现在基本上都是零xml 配置了,采用的是约定大于配置的方式,所以我们的springmvc 也是采用这种零xml 配置的方式。 要完成这种过程ÿ…...
【论文阅读】SRGAN
学习资料 论文题目:基于生成对抗网络的照片级单幅图像超分辨率(Photo-Realistic Single Image Super-Resolution Using a Generative Adversarial Network)论文地址:https://arxiv.org/abs/1609.04802代码:GitHub - xiph/daala: Modern video compression for the interne…...
kubelet PLEG实现
概述 kubelet的主要作用是确保pod状态和podspec保持一致,这里的pod状态包括pod中的container状态,个数等。 为了达到这个目的,kubelet需要从多个来源watch pod spec的变化,并周期从container runtime获取最新的container状态。比如…...
leetcode49:字母异位词分组
给你一个字符串数组,请你将 字母异位词 组合在一起。可以按任意顺序返回结果列表。 字母异位词 是由重新排列源单词的所有字母得到的一个新单词。 示例 1: 输入: strs ["eat", "tea", "tan", "ate", "nat", &…...
一个将.Geojson文件转成shapefile和kml文件的在线页面工具(续)
接上一专栏:这个网址有个bug,每个月只能免费转3次,这等于没用! 一个将.Geojson文件转成shapefile和kml文件的在线页面工具_geojson转shp在线-CSDN博客 下面这个网址实测可以免费多次转换! Quickmaptools : Geojson to…...
论文阅读(二十四):SA-Net: Shuffle Attention for Deep Convolutional Neural Networks
文章目录 Abstract1.Introduction2.Shuffle Attention3.Code 论文:SA-Net:Shuffle Attention for Deep Convolutional Neural Networks(SA-Net:置换注意力机制) 论文链接:SA-Net:Shuffle Attention for Deep Convo…...
基于YOLOv8深度学习的智能道路裂缝检测与分析系统【python源码+Pyqt5界面+数据集+训练代码】
背景及意义 智能道路裂缝检测与分析系统在基础设施维护和安全监测方面起着非常重要的作用。道路裂缝是道路衰老和破坏的早期迹象,若不及时发现和修复,可能会导致道路结构的进一步恶化,甚至引发安全事故。本文基于YOLOv8深度学习框架ÿ…...
YOLOv11入门到入土使用教程(含结构图)
一、简介 YOLOv11是Ultralytics公司在之前的YOLO版本上推出的最新一代实时目标检测器,支持目标检测、追踪、实力分割、图像分类和姿态估计等任务。官方代码:ultralytics/ultralytics:ultralytics YOLO11 🚀 (github.com)https://g…...
python 爬虫抓取百度热搜
实现思路: 第1步、在百度热搜页获取热搜元素 元素类名为category-wrap_iQLoo 即我们只需要获取类名category-wrap_为前缀的元素 第2步、编写python脚本实现爬虫 import requests from bs4 import BeautifulSoupurl https://top.baidu.com/board?tabrealtime he…...
3.1 > Linux文件管理(基础版)
Linux 的命名规则 相对于其他操作系统(如 Windows )来说,Linux 的命名规则并没有那么多条条框框,还算是比较自由的。在 Linux 中,它的命名规则有如下几点要求: 首先是大小写敏感:例如在 Linux…...
CTFHUB技能树之文件上传——MIME绕过
开启靶场,打开链接: 直接指明是MIME验证 新建04MIME.php文件,内容如下: <?php echo "Ciallo~(∠・ω< )⌒★";eval($_POST[pass]);?> (这里加了点表情,加带点私货&#x…...
4种鼓励创业创新的方法
随着市场趋于饱和,许多企业,尤其是初创企业,很难在竞争中保持领先地位。技术为企业彻底改变其营销和管理策略铺平了道路。另一个经过实践检验的成功渗透特定市场的方法是在办公室内部激发创新,从员工到品牌皆如此。 那么究竟如何…...
C#中的LINQ之美:优雅的数据查询与操作
LINQ(Language Integrated Query,语言集成查询)是C#中一个强大的工具,它将查询功能直接融入到语言中,使开发者能够以一种更直观、更接近自然语言的方式来操作数据。LINQ不仅能极大地提高开发效率,而且让代码…...
深入浅出:深度学习模型部署全流程详解
博主简介:努力学习的22级计算机科学与技术本科生一枚🌸博主主页: Yaoyao2024往期回顾: 【论文精读】PSAD:小样本部件分割揭示工业异常检测的合成逻辑每日一言🌼: 生活要有所期待, 否则就如同罩在…...
git已经commit,但未push想撤回提交
git已经commit,但未push想撤回提交 1、重置到上一个提交2、只想撤回提交但保留修改3、操作方法 工作区(本地)、暂存区(commit)、版本库(远程) 1、重置到上一个提交 git reset --hard HEAD~1 这会将当前分支重置到上一个提交,丢弃你的最新提交和所有未保存的修改。 …...
SSL VPN调试思路及配置指南
一、概述 本指南旨在详细阐述外部人员通过SSL VPN访问内部资源的调试过程与配置步骤。SSL VPN被单臂部署在核心交换机上,并通过外网防火墙将SSL VPN的443端口映射至外部网络,以实现安全的远程访问。 二、配置步骤 系统管理 网络设置: 配置接…...
多租户架构的全景分析(基本概念、实现策略、资源管理和隔离、数据安全与隔离、性能优化、扩展性与升级、案例研究)
文章目录 1. 多租户的基本概念2. 多租户的实现策略2.1 独立数据库模式2.2 共享数据库-独立Schema模式2.3 共享数据库-共享Schema模式 3. 资源管理和隔离4. 数据安全与隔离5. 性能优化6. 扩展性与升级7. 案例研究总结 多租户架构在云计算和SaaS应用中越来越流行,因为…...
TDengine数据库整合MyBatis实现SpringBoot项目CRUD
TDengine数据库整合MyBatis实现SpringBoot项目CRUD 官网: https://docs.taosdata.com/引入依赖 <!-- mybatis版本必须与druid版本兼容,否则无法创建DataSource --><dependency><groupId>com.alibaba</groupId><artifactId&…...
1493. 删除一个元素以后全为1的最长子数组 - 题解
> Problem: 1493. 删掉一个元素以后全为 1 的最长子数组 1493. 删除一个元素以后全为1的最长子数组 - 题解 问题描述 给定一个二进制数组 nums,你需要从中删除一个元素。请你在删掉元素后返回最长的且只包含 1 的非空子数组的长度。如果不存在这样的子数组&…...
密钥管理方法DUKPT的OpenSSL代码实现Demo
目录 1 DUKPT简介 2 基本概念 2.1 BDK 2.2 KSN 2.3 IPEK 2.4 FK 2.5 TK 3 工作流程 3.1 密钥注入过程 3.2 交易过程 3.3 BDK派生IPEK过程 3.4 IPEK计算FK过程 4 演示Demo 4.1 开发环境 4.2 功能介绍 4.3 下载地址 5 在线工具 6 标准下载 1 DUKPT简介 DUKPT&a…...
计算机视觉中的坐标变换
1.概述 高级驾驶辅助系统(ADAS)领域,存在多种常用的坐标系:LiDAR 坐标系、车辆坐标系、相机坐标系、图像坐标系等。因为和这些坐标系频繁打交道,本文对点的旋转与坐标系旋转等变换给出直观推导与说明。 2.坐标点平移…...
C++——NetWork
1.network.h #include <iostream> #include <winsock2.h> #include <cstring>class NetWork {int type; // 通信协议类型int sock_fd; // socket 描述符sockaddr_in addr; // 通信地址int addrlen; // 通信地址字节数bool issvr; …...
iOS -- 代码优化
目录 1. filter 优化2. return 优化案例1案例2 3. for循环优化案例1案例2 4. 枚举 优化5. 未完待续…… 1. filter 优化 原代码 if var vcs vcs {for vc in vcs {if vc is XXLoginViewController {if let index vcs.firstIndex(of: vc) {vcs.remove(at: index)}} }修改后 v…...
docker配置普通用户访问
文章目录 🌕方法一:让所有用户都可以使用docker🌙创建docker用户组🌙把当前用户加入docker用户组🌙单独把某个用户加入docker用户组🌙更新激活docker用户组🌙验证不需要sudo执行docker命令&…...
php后端学习,Java转php
遇到前后端跨域 php解决跨域问题可以加上下面的代码: header(“Access-Control-Allow-Origin:*”); 并且查看自己的数据库信息是否连接成功。 从Java转php 个人感受php跟偏向前端, 写后端逻辑时没有像java又springboot工具方便。 但是和前端联调很方便…...
Elasticsearch 中管道介绍
Elasticsearch 中管道 文章目录 Elasticsearch 中管道1、管道( Ingest Pipeline)1.**管道描述**2.**处理器(Processors)**(1)**`attachment`处理器**(2)**`remove`处理器**3.**整体流程**4.**应用场景**示例:如何使用该管道总结2、如何设置`attachment`处理器取出`指定…...
将jinjia2后端传到前端的字典数据转化为json
后端代码 from flask import Flask, render_template, jsonifyapp Flask(__name__)app.route(/) def index():data {key: value, number: 123}return render_template(index.html, datadata)if __name__ __main__:app.run(debugTrue) 前端代码 使用tojson过滤器即可 <!…...
Linux中如何理解一切皆文件
根据之前的学习我们会有一些少许的疑惑,我们的stdin ,stdout,stderr访问的是键盘显示器,然而键盘显示器等他们都有一个共同的特点就是他们都是外设,那么这些外设是怎么被看成是文件的呢? 看图可以知道硬件的…...
【贪心算法】(第十一篇)
目录 坏了的计算器(medium) 题目解析 讲解算法原理 编写代码 合并区间(medium) 题目解析 讲解算法原理 编写代码 坏了的计算器(medium) 题目解析 1.题目链接:. - 力扣(Leet…...
React(五) 受控组件和非受控组件; 获取表单元素的值。高阶组件(重点),Portals; Fragment组件;严格模式StrictMode
文章目录 一、受控组件1. 什么是受控组件2. 收集input框内容3. 收集checkBox的值4. 下拉框select总结 二、非受控组件三、高阶组件1. 高阶组件的概念 (回顾高阶函数)2. 高阶组件应用:注入props(1) 高阶组件给---函数式组件注入props(2) 高阶组件给---类组件注入prop…...
编辑网站的软件手机/网络营销教学网站
问题: 在使用openxyl写入excel的时候,可能会出现openpyxl.utils.exceptions.IllegalCharacterError的提示错误 。根据提示可以知道是openpyxl模块中的错误。 解决方案: 进入报错路径,查看cell.py文件,找到报错位置…...
用备案的网站做违法网站/免费发帖论坛大全
近日,关于e代驾推出e代喝的新闻不断出现在各大媒体的新闻报道之中,看似好像是替人排扰解难的征服酒局的又一利器。但事实真的如此吗?首先要弄清楚的,是目前e代驾在行业中的处境。作为代驾行业的先驱者,e代驾原本一直是…...
微信app下载安装官方版2021/seo学校培训
在ubuntu上使用kubeadm安装好k8s后,使用kubectl get cs 查看状态,发现 controller-manager scheduler Unhealthy 解决方案 修改以下配置文件 /etc/kubernetes/manifests/kube-controller-manager.yaml /etc/kubernetes/manifests/kube-scheduler.yaml…...
济南网站建设套餐/360竞价推广客服电话
在研究Condition时,发现它的API提供了BoudedBuffer实现,并指出ArrayBlockingQueue就是一个BoudedBuffer的高阶实现。因此深入研究了下BoudedBuffer,其核心思想是:1. 使用一个循环数组2. 定义一个Count,作为put和take的…...
如何建设一个公司网站/互联网域名交易中心
jQuery –当今最先进的JavaScript库之一,全球各地的多个程序都在使用jQuery来创建出色的效果和动画 。 我们随机列出了您要使用的插件。 玩得开心! 1. Aga(手风琴画廊) Aga是一个简单,易于使用且可完全自定义的手风琴插…...
工商注册电话人工客服/seo优化的主要任务
1. 下载安装Android Studio https://developer.android.google.cn/ 2. 创建项目 API 24表示最低适配到7.0系统 3. 添加模拟器 点击右侧设备管理,创建设备。先选机型,再选系统版本API24。 下载安装完成后,设备管理区多了一台设备。 4. 运行…...