Spring核心方法:Refresh全解(WebMVC如何装配、关联)
Spring核心方法:Refresh全解(WebMVC如何装配、关联)
这里是一个表格,列出了Spring容器刷新过程中执行的方法以及它们的作用:
| 方法名称 | 描述 |
|---|---|
| prepareRefresh() | 初始化一些属性和状态,例如启动时间戳、活动标志、环境变量等。 |
| obtainFreshBeanFactory() | 创建一个新的Bean Factory,并配置一些基本属性。 |
| prepareBeanFactory() | 对Bean Factory进行进一步的配置。 |
| postProcessBeanFactory() | 允许子类对Bean Factory进行后处理。 |
| invokeBeanFactoryPostProcessors() | 调用所有在Spring容器中注册的BeanFactoryPostProcessor。 |
| registerBeanPostProcessors() | 注册所有在Spring容器中注册的Bean后处理器。 |
| initMessageSource() | 初始化Spring容器中的消息源。 |
| initApplicationEventMulticaster() | 初始化Spring容器中的事件多路广播器。 |
| onRefresh() | 允许子类在Spring容器初始化完成后进行一些自定义的初始化操作。 |
| registerListeners() | 注册所有在Spring容器中注册的监听器。 |
| finishBeanFactoryInitialization() | 初始化所有在Spring容器中注册的单例Bean。 |
| finishRefresh() | 发布一个刷新完成事件,表示Spring容器已经成功初始化并准备就绪。 |
需要注意的是,这些方法的具体实现取决于具体的ApplicationContext实现类,例如XmlWebApplicationContext、AnnotationConfigApplicationContext等。因此,在具体使用时需要结合具体的实现类来学习和理解这些方法。
Spring容器刷新过程详解
Spring容器在初始化过程中会执行一系列的操作,以确保容器中的各种组件都正确地创建和配置。这个过程被称为容器刷新,它由AbstractApplicationContext类的refresh()方法触发。以下是Spring容器刷新过程中执行的方法以及它们的作用:
好的,现在让我来整理一下您之前提到的四个问题,并结合源码对refresh()方法以及牵扯到的其他方法进行详细的中文解释。
-
Spring容器刷新方法概述
Spring容器刷新是指在Spring容器初始化时,创建和配置Bean Factory,并初始化Spring容器中的各种组件的过程。refresh()方法是AbstractApplicationContext类中用于刷新Spring容器的主要方法。该方法在容器初始化时被调用,用于创建和配置Bean Factory,并初始化Spring容器中的各种组件。
-
refresh()方法源码解释
以下是refresh()方法的源码,逐行解释如下:
@Override public void refresh() throws BeansException, IllegalStateException {synchronized (this.startupShutdownMonitor) {// 1. 准备刷新prepareRefresh();// 2. 获取新的Bean FactoryConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();// 3. 准备Bean FactoryprepareBeanFactory(beanFactory);try {// 4. 允许子类对Bean Factory进行后处理postProcessBeanFactory(beanFactory);// 5. 调用BeanFactoryPostProcessorinvokeBeanFactoryPostProcessors(beanFactory);// 6. 注册Bean后处理器registerBeanPostProcessors(beanFactory);// 7. 初始化MessageSourceinitMessageSource();// 8. 初始化事件多路广播器initApplicationEventMulticaster();// 9. 其他特殊Bean的初始化工作onRefresh();// 10. 注册监听器registerListeners();// 11. 初始化所有剩余的单例BeanfinishBeanFactoryInitialization(beanFactory);// 12. 发布刷新完成事件finishRefresh();}catch (BeansException ex) {if (logger.isWarnEnabled()) {logger.warn("Exception encountered during context initialization - " +"cancelling refresh attempt: " + ex);}// 销毁已创建的BeandestroyBeans();// 重置活动标志cancelRefresh(ex);// 抛出异常throw ex;}finally {// 重置公共缓存resetCommonCaches();}} }
以上是refresh()方法的源码,接下来我们逐步解释每一步的具体作用。
- prepareRefresh()方法源码解释
prepareRefresh()方法是refresh()方法的第一步,主要用于初始化一些属性和状态,例如启动时间戳、活动标志、环境变量等。以下是prepareRefresh()方法的源码,逐行解释如下:
protected void prepareRefresh() {// 1. 切换到活动状态this.startupDate = System.currentTimeMillis();this.closed.set(false);this.active.set(true);if (logger.isDebugEnabled()) {logger.debug("Refreshing " + this);}// 2. 初始化属性源initPropertySources();// 3. 验证所有必需的属性是否可解析getEnvironment().validateRequiredProperties();// 4. 存储预刷新的ApplicationListenersif (this.earlyApplicationListeners != null) {this.applicationListeners.addAll(this.earlyApplicationListeners);this.earlyApplicationListeners = null;}// 5. 允许收集早期的ApplicationEvents,等待多路广播器可用时发布...this.earlyApplicationEvents = new LinkedHashSet<>(this.applicationEvents);this.applicationEvents = null;// 6. 创建并配置新的ApplicationEventMulticasterthis.eventMulticaster = new SimpleApplicationEventMulticaster(this);// 7. 添加某些系统监听器this.eventMulticaster.setTaskExecutor(this.taskExecutor);this.eventMulticaster.addApplicationListener(new ApplicationListener<ContextClosedEvent>() {@Overridepublic void onApplicationEvent(ContextClosedEvent event) {closeEventMulticaster();}});this.eventMulticaster.addApplicationListener(new ApplicationListener<ContextRefreshedEvent>() {@Overridepublic void onApplicationEvent(ContextRefreshedEvent event) {initApplicationEventListeners();}});this.eventMulticaster.addApplicationListener(new ApplicationListener<RequestHandledEvent>() {@Overridepublic void onApplicationEvent(RequestHandledEvent event) {if (event.getException() != null && shouldPublishEvent(event.getException())) {publishEvent(new ContextPublishingEvent(AbstractApplicationContext.this, event));}}});// 8. 发布早期的ApplicationEvents,现在我们终于有了多路广播器...Set<ApplicationEvent> earlyEventsToProcess = this.earlyApplicationEvents;this.earlyApplicationEvents = null;for (ApplicationEvent earlyEvent : earlyEventsToProcess) {this.eventMulticaster.multicastEvent(earlyEvent);}// 9. 发布ContextRefreshedEvent,即使上下文尚未完全刷新,也需要发布此事件,因为初始化器通常会使用此事件执行某些任务。this.eventMulticaster.multicastEvent(new ContextRefreshedEvent(this));// 10. 初始化生命周期处理器initLifecycleProcessor();// 11. 首先将刷新传播到生命周期处理器getLifecycleProcessor().onRefresh();// 12. 发布最终事件finishRefresh();
}
以上是prepareRefresh()方法的源码,接下来我们逐步解释每一步的具体作用。
- SpringMVC通过refresh()方法关联Spring容器
在SpringMVC中,DispatcherServlet是整个Web应用的前端控制器,它负责接收所有的请求并将请求分发到相应的处理器中。在DispatcherServlet初始化时,会创建一个Spring容器,并通过refresh()方法进行初始化。以下是DispatcherServlet中关于refresh()方法的部分源码:
protected WebApplicationContext initWebApplicationContext() {WebApplicationContext wac = null;try {// 1. 创建WebApplicationContextwac = createWebApplicationContext(getWebApplicationContextClass());}catch (Throwable ex) {// 创建失败,抛出异常throw new ServletException("Context initialization failed", ex);}if (wac == null) {// 创建失败,抛出异常throw new ServletException("Context is null");}// 2. 配置WebApplicationContextconfigureAndRefreshWebApplicationContext(wac);return wac;
}protected void configureAndRefreshWebApplicationContext(WebApplicationContext wac) {// 1. 配置WebApplicationContextwac.setServletContext(getServletContext());wac.setServletConfig(getServletConfig());wac.setNamespace("dispatcher-servlet");wac.addApplicationListener(new ContextLoaderListener(wac));// 2. 刷新WebApplicationContextwac.refresh();
}
可以看到,在DispatcherServlet中,首先通过createWebApplicationContext()方法创建一个WebApplicationContext对象,然后通过configureAndRefreshWebApplicationContext()方法对其进行配置和刷新。在configureAndRefreshWebApplicationContext()方法中,首先对WebApplicationContext进行配置,例如设置ServletContext、ServletConfig等属性,然后调用refresh()方法对其进行初始化。这样,SpringMVC就通过DispatcherServlet将Spring容器关联到了Web应用中。
- 结论
本文主要介绍了Spring容器刷新的概述、refresh()方法的源码解释、prepareRefresh()方法的源码解释、以及SpringMVC通过refresh()方法关联Spring容器的具体过程。通过对这些方法的详细解释,希望能够帮助大家更好地理解Spring容器的刷新过程以及SpringMVC中的应用。
详细解释
接下来,我们将详细解释Spring容器刷新过程中执行的每个方法,以帮助您更好地理解Spring容器的工作原理。
prepareRefresh()
prepareRefresh()方法是Spring容器刷新过程中的第一个方法。在这个方法中,Spring容器会执行以下操作:
-
设置容器的启动时间、活动标志和环境变量等属性。
-
初始化容器的属性源,包括系统属性、环境变量和配置文件等。
-
创建一个新的ApplicationEventMulticaster,用于广播容器中的事件。
-
注册一些默认的监听器,例如ContextRefreshedEvent、ContextClosedEvent等。
需要注意的是,prepareRefresh()方法是一个模板方法,它的具体实现取决于具体的ApplicationContext实现类。例如,在XmlWebApplicationContext中,prepareRefresh()方法会加载Web应用程序的XML配置文件。
obtainFreshBeanFactory()
obtainFreshBeanFactory()方法用于创建一个新的BeanFactory,并配置一些基本属性。在这个方法中,Spring容器会执行以下操作:
-
创建一个新的DefaultListableBeanFactory,它是Spring容器中BeanFactory的默认实现。
-
设置BeanFactory的ClassLoader、PropertyEditor注册器等基本属性。
-
注册一些默认的Bean,例如ApplicationContext、ApplicationEventMulticaster等。
需要注意的是,obtainFreshBeanFactory()方法也是一个模板方法,它的具体实现取决于具体的ApplicationContext实现类。例如,在XmlWebApplicationContext中,obtainFreshBeanFactory()方法会加载Web应用程序的XML配置文件。
相关文章:
Spring核心方法:Refresh全解(WebMVC如何装配、关联)
Spring核心方法:Refresh全解(WebMVC如何装配、关联) 这里是一个表格,列出了Spring容器刷新过程中执行的方法以及它们的作用: 方法名称描述prepareRefresh()初始化一些属性和状态,例如启动时间戳、活动标志、环境变量等。obtainF…...
TCP:三次握手四次挥手及相关问题:
连接—三次握手: 流程图: 过程详解: 客户端(connect)连接服务器(listen) Client将标志位SYN置为1,随机产生一个值seqx, 并将该数据包发送给Server, Client进入SYN_ SENT状态,等待Server确认。Server收到数据包后由标…...
链式二叉树--前序中序后序遍历,高度,节点个数问题
目录 前言: 一:链式二叉树的结构定义 二:链式二叉树的遍历--->前序,中序,后序 1.前序 递归展开图分析 2.中序 递归展开图分析 3.后序 三:二叉树结点的求解 1.二叉树总结点 递归展开分析 2…...
HCIA——TCP协议详解
目录 1、TCP概念及协议头部格式 1.1TCP特点 1.2TCP协议协议头部格式 1.3字段进行介绍 1.3.1源端口和目的端口 1.3.2序号(seq) 1.3.3确认序号(ack) 1.3.4数据偏移 1.3.5标志位 1.3.6窗口 1.3.7校验和 1.3.8紧急指针 2、TCP的可靠性 2.1 TCP可靠性的保障 2.2排序机…...
Hadoop大数据应用:Linux 部署 HDFS 分布式集群
目录 一、实验 1.环境 2.Linux 部署 HDFS 分布式集群 3.Linux 使用 HDFS 文件系统 二、问题 1.ssh-copy-id 报错 2. 如何禁用ssh key 检测 3.HDFS有哪些配置文件 4.hadoop查看版本报错 5.启动集群报错 6.hadoop 的启动和停止命令 7.上传文件报错 8.HDFS 使用命令 一…...
纯 CSS 实现文字换行环绕效果
实现效果 实现代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title>Document</title><…...
【爬虫逆向】Python逆向采集猫眼电影票房数据
进行数据抓包,因为这个网站有数据加密 !pip install jsonpathCollecting jsonpathDownloading jsonpath-0.82.2.tar.gz (10 kB)Preparing metadata (setup.py) ... done Building wheels for collected packages: jsonpathBuilding wheel for jsonpath (setup.py) .…...
解析服务器下载速度:上行、下行与带宽之谜
在日常使用中,我们经常会遇到从服务器下载内容速度忽快忽慢的情况,即便服务器的硬件配置如4核CPU、8GB内存和12Mbps的带宽看似足够。为何会出现这种现象?这背后涉及到网络中的上行、下行以及带宽等关键概念。本文旨在揭开这些术语背后的含义&…...
计算机网络的概念
目录 <计算机网络的定义> <计算机网络的形成与发展> 1.第一阶段远程联机阶段----60年代以前: 2.第二阶段多机互联网络阶段----60年代中期: 3.第三阶段标准化网络阶段----70年代末: 4.第四阶段网络互联与高速网络阶段一90年代: <计算机网络的未来--下一代…...
MATLAB中的脚本和函数有什么区别?
MATLAB中的脚本和函数是两种不同的代码组织方式,它们在结构、功能和使用方式上有显著的区别。以下是对这两种方式的详细解释,总计约2000字。 一、MATLAB脚本 MATLAB脚本是一种包含多条MATLAB命令的文件,这些命令按照在文件中的顺序依次执行…...
从电影《沙丘》说起——对人工智能的思考
正文 从《沙丘》开始说起 之前看《沙丘》电影,里面有一类角色叫门泰特,这类人大脑可以飞快地运算,在电影设定里是替换人工智能、机器运算的存在。男主保罗也是这类型的人,但他可能基因更强大,吸食了香料后࿰…...
使用Python进行自然语言处理(NLP):NLTK与Spacy的比较【第133篇—NLTK与Spacy】
👽发现宝藏 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。【点击进入巨牛的人工智能学习网站】。 使用Python进行自然语言处理(NLP):NLTK与Spacy的比较 自…...
学习笔记--在线强化学习与离线强化学习的异同(3)
这篇博文很多部分仅代表个人学习观点,欢迎大家与我一起讨论 强化学习与离线强化学习的区别 强化学习和离线强化学习都是机器学习的分支,主要用于训练智能体以在不断尝试和错误的过程中学习如何最大化累积奖励。它们之间的主要区别在于数据的获取方式和训…...
使用Thymeleaf导出PDF,页眉插入图片与内容重叠?
CSS 打印分页功能 需求:打印 在第一页的内容被挤到第二页的时候,又想每一页页头都有相同的样式,使用页眉。 问题:第二页的内容与页眉重叠了? 查各路找出的原因:header 页眉不占空间 解决:不…...
python网络编程:通过socket实现TCP客户端和服务端
目录 写在开头 socket服务端(基础) socket客户端(基础) 服务端实现(可连接多个客户端) 客户端实现 数据收发效果 写在开头 近期可能会用python实现一些网络安全工具,涉及到许多关于网络…...
论文阅读——RSGPT
RSGPT: A Remote Sensing Vision Language Model and Benchmark 贡献:构建了一个高质量的遥感图像描述数据集(RSICap)和一个名为RSIEval的基准评估数据集,并在新创建的RSICap数据集上开发了基于微调InstructBLIP的遥感生成预训练…...
长连接技术
个人学习记录,欢迎指正 1.轮询 1.1 轮询的形式 短连接轮询 前端每隔一段时间向服务端发起一次Http请求来获取数据。 const shortPolling () > { const intervalHandler setInterval(() > {fetch(/xxx/yyy).then(response > response.json()).then(respo…...
供电系统分类详解
一、供电系统分类 电力供电系统一般有5种供电模式,常用的有:IT系统,TT系统,TN系统,其中TN系统又可以分为TN-C,TN-S,TN-C-S。 1、TN-C系统(三相四线制) 优点: 该系统中…...
基于centos7的k8s最新版v1.29.2安装教程
k8s概述 Kubernetes 是一个可移植、可扩展的开源平台,用于管理容器化的工作负载和服务,可促进声明式配置和自动化。 Kubernetes 拥有一个庞大且快速增长的生态,其服务、支持和工具的使用范围相当广泛。 Kubernetes 这个名字源于希腊语&…...
【赠书第20期】AI绘画与修图实战:Photoshop+Firefly从入门到精通
文章目录 前言 1 入门篇:初识Photoshop与Firefly 2 进阶篇:掌握Photoshop与Firefly的核心技巧 3 实战篇:运用Photoshop与Firefly进行创作 4 精通篇:提升创作水平,拓展应用领域 5 结语 6 推荐图书 7 粉丝福利 前…...
【Axure高保真原型】引导弹窗
今天和大家中分享引导弹窗的原型模板,载入页面后,会显示引导弹窗,适用于引导用户使用页面,点击完成后,会显示下一个引导弹窗,直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…...
工程地质软件市场:发展现状、趋势与策略建议
一、引言 在工程建设领域,准确把握地质条件是确保项目顺利推进和安全运营的关键。工程地质软件作为处理、分析、模拟和展示工程地质数据的重要工具,正发挥着日益重要的作用。它凭借强大的数据处理能力、三维建模功能、空间分析工具和可视化展示手段&…...
网络编程(UDP编程)
思维导图 UDP基础编程(单播) 1.流程图 服务器:短信的接收方 创建套接字 (socket)-----------------------------------------》有手机指定网络信息-----------------------------------------------》有号码绑定套接字 (bind)--------------…...
HashMap中的put方法执行流程(流程图)
1 put操作整体流程 HashMap 的 put 操作是其最核心的功能之一。在 JDK 1.8 及以后版本中,其主要逻辑封装在 putVal 这个内部方法中。整个过程大致如下: 初始判断与哈希计算: 首先,putVal 方法会检查当前的 table(也就…...
Java毕业设计:WML信息查询与后端信息发布系统开发
JAVAWML信息查询与后端信息发布系统实现 一、系统概述 本系统基于Java和WML(无线标记语言)技术开发,实现了移动设备上的信息查询与后端信息发布功能。系统采用B/S架构,服务器端使用Java Servlet处理请求,数据库采用MySQL存储信息࿰…...
DingDing机器人群消息推送
文章目录 1 新建机器人2 API文档说明3 代码编写 1 新建机器人 点击群设置 下滑到群管理的机器人,点击进入 添加机器人 选择自定义Webhook服务 点击添加 设置安全设置,详见说明文档 成功后,记录Webhook 2 API文档说明 点击设置说明 查看自…...
【Linux手册】探秘系统世界:从用户交互到硬件底层的全链路工作之旅
目录 前言 操作系统与驱动程序 是什么,为什么 怎么做 system call 用户操作接口 总结 前言 日常生活中,我们在使用电子设备时,我们所输入执行的每一条指令最终大多都会作用到硬件上,比如下载一款软件最终会下载到硬盘上&am…...
【1】跨越技术栈鸿沟:字节跳动开源TRAE AI编程IDE的实战体验
2024年初,人工智能编程工具领域发生了一次静默的变革。当字节跳动宣布退出其TRAE项目(一款融合大型语言模型能力的云端AI编程IDE)时,技术社区曾短暂叹息。然而这一退场并非终点——通过开源社区的接力,TRAE在WayToAGI等…...
RK3568项目(七)--uboot系统之外设与PMIC详解
目录 一、引言 二、按键 ------>2.1、按键种类 ------------>2.1.1、RESET ------------>2.1.2、UPDATE ------------>2.1.3、PWRON 部分 ------------>2.1.4、RK809 PMIC ------------>2.1.5、ADC按键 ------------>2.1.6、ADC按键驱动 ------…...
C++核心编程_继承同名成员处理方式
问题:当子类与父类出现同名的成员,如何通过子类对象,访问到子类或父类中同名的数据呢? 访问子类同名成员 直接访问即可 访问父类同名成员 需要加作用域 class Base { public:Base(){m_A 100;}void func(){cout << "B…...
