spring ioc源码-refresh();

2024-01-01 04:04
文章标签 java 源码 spring ioc refresh

本文主要是介绍spring ioc源码-refresh();,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

主要作用是刷新应用上下文

@Override
public void refresh() throws BeansException, IllegalStateException {synchronized (this.startupShutdownMonitor) {// 启动刷新的性能跟踪步骤StartupStep contextRefresh = this.applicationStartup.start("spring.context.refresh");// 准备刷新,设置标志位,保存了容器的启动时间prepareRefresh();// 子类实现,获取一个新的 BeanFactoryConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();// 准备 BeanFactory,设置一些基本的配置prepareBeanFactory(beanFactory);try {// 允许子类对 BeanFactory 进行进一步的处理postProcessBeanFactory(beanFactory);// 启动性能跟踪步骤,执行在容器中注册的 BeanFactoryPostProcessorsStartupStep beanPostProcess = this.applicationStartup.start("spring.context.beans.post-process");invokeBeanFactoryPostProcessors(beanFactory);// 注册 BeanPostProcessors,这些 processors 会拦截 Bean 的创建过程registerBeanPostProcessors(beanFactory);beanPostProcess.end();// 初始化消息源initMessageSource();// 初始化事件广播器initApplicationEventMulticaster();// 子类实现,初始化一些特殊的 beansonRefresh();// 注册监听器 beansregisterListeners();// 实例化所有剩余的(非懒加载的)单例 beansfinishBeanFactoryInitialization(beanFactory);// 最后一步:发布相应的事件finishRefresh();} catch (BeansException ex) {if (logger.isWarnEnabled()) {logger.warn("Exception encountered during context initialization - " +"cancelling refresh attempt: " + ex);}// 销毁已创建的单例 beans,以避免悬空资源destroyBeans();// 重置 'active' 标志位cancelRefresh(ex);// 将异常传播给调用者throw ex;} finally {// 重置 Spring 核心中的常见内省缓存,因为我们可能不再需要单例 beans 的元数据了resetCommonCaches();// 结束性能跟踪步骤contextRefresh.end();}}
}

主要就做了这些,对上面的方法进行分解:

prepareRefresh():

主要就是保存了启动时间,启动标志

protected void prepareRefresh() {// Switch to active.this.startupDate = System.currentTimeMillis();this.closed.set(false);this.active.set(true);//....省略部分代码}
prepareBeanFactory  
做准备工作
protected void prepareBeanFactory(ConfigurableListableBeanFactory beanFactory) {// Tell the internal bean factory to use the context's class loader etc.//设置类加载器		
beanFactory.setBeanClassLoader(getClassLoader());if (!shouldIgnoreSpel) {beanFactory.setBeanExpressionResolver(new StandardBeanExpressionResolver(beanFactory.getBeanClassLoader()));}beanFactory.addPropertyEditorRegistrar(new ResourceEditorRegistrar(this, getEnvironment()));// Configure the bean factory with context callbacks.//添加后置处理器:ApplicationContextAwareProcessorbeanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this));//忽略自动装配beanFactory.ignoreDependencyInterface(EnvironmentAware.class);beanFactory.ignoreDependencyInterface(EmbeddedValueResolverAware.class);beanFactory.ignoreDependencyInterface(ResourceLoaderAware.class);beanFactory.ignoreDependencyInterface(ApplicationEventPublisherAware.class);beanFactory.ignoreDependencyInterface(MessageSourceAware.class);beanFactory.ignoreDependencyInterface(ApplicationContextAware.class);beanFactory.ignoreDependencyInterface(ApplicationStartupAware.class);// BeanFactory interface not registered as resolvable type in a plain factory.//以下接口,允许自动装配,第一个参数是自动装配的类型,,第二个字段是自动装配的值// MessageSource registered (and found for autowiring) as a bean.beanFactory.registerResolvableDependency(BeanFactory.class, beanFactory);beanFactory.registerResolvableDependency(ResourceLoader.class, this);beanFactory.registerResolvableDependency(ApplicationEventPublisher.class, this);beanFactory.registerResolvableDependency(ApplicationContext.class, this);// Register early post-processor for detecting inner beans as //添加一个后置处理器:ApplicationListenerDetector,此后置处理器实现了BeanPostProcessor接口
ApplicationListeners.beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(this));// Detect a LoadTimeWeaver and prepare for weaving, if found.if (!NativeDetector.inNativeImage() && beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) {beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory));// Set a temporary ClassLoader for type matching.beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader()));}// Register default environment beans.//在容器中还没有XX的bean的时候,帮我们注册beanName为XX的singleton beanif (!beanFactory.containsLocalBean(ENVIRONMENT_BEAN_NAME)) {beanFactory.registerSingleton(ENVIRONMENT_BEAN_NAME, getEnvironment());}if (!beanFactory.containsLocalBean(SYSTEM_PROPERTIES_BEAN_NAME)) {beanFactory.registerSingleton(SYSTEM_PROPERTIES_BEAN_NAME, getEnvironment().getSystemProperties());}if (!beanFactory.containsLocalBean(SYSTEM_ENVIRONMENT_BEAN_NAME)) {beanFactory.registerSingleton(SYSTEM_ENVIRONMENT_BEAN_NAME, getEnvironment().getSystemEnvironment());}if (!beanFactory.containsLocalBean(APPLICATION_STARTUP_BEAN_NAME)) {beanFactory.registerSingleton(APPLICATION_STARTUP_BEAN_NAME, getApplicationStartup());}}

invokeBeanFactoryPostProcessors(beanFactory);

主要逻辑代码在org.springframework.context.support.PostProcessorRegistrationDelegate#invokeBeanFactoryPostProcessors(org.springframework.beans.factory.config.ConfigurableListableBeanFactory, java.util.List<org.springframework.beans.factory.config.BeanFactoryPostProcessor>)

代码设计太多就不粘贴了,主要做了一下逻辑:

  1. 初始化设置:

    • 该方法接收一个 ConfigurableListableBeanFactory 和一个 BeanFactoryPostProcessor 实例的列表。
  2. 处理 BeanDefinitionRegistryPostProcessor 实例:

    • 如果 beanFactory 也是 BeanDefinitionRegistry(用于管理 bean 定义的扩展接口),则该方法区分常规的 BeanFactoryPostProcessors 和实现了 BeanDefinitionRegistryPostProcessor 的处理器。
    • 它遍历提供的 beanFactoryPostProcessors,对于实现了 BeanDefinitionRegistryPostProcessor 接口的实例,调用其 postProcessBeanDefinitionRegistry 方法,并将它们收集到不同的列表中。
  3. 按顺序处理 BeanDefinitionRegistryPostProcessor

    • 该方法按照三个阶段处理 BeanDefinitionRegistryPostProcessor:PriorityOrdered、Ordered 和其他处理器。
    • 对于每个阶段,它获取相关的后置处理器名称,实例化它们,根据它们的优先级或顺序排序,并调用它们的 postProcessBeanDefinitionRegistry 方法。
  4. 处理已注册的 BeanFactoryPostProcessor 实例:

    • 如果 beanFactory 不是 BeanDefinitionRegistry,则直接调用已注册的 BeanFactoryPostProcessor 实例。
  5. 按顺序处理 BeanFactoryPostProcessor

    • 该方法根据实现 PriorityOrderedOrdered 和其他接口的 BeanFactoryPostProcessor 分别处理。
    • 首先,调用实现了 PriorityOrdered 接口的后置处理器。
    • 接着,调用实现了 Ordered 接口的后置处理器。
    • 最后,调用所有其他 BeanFactoryPostProcessor
  6. 清除缓存的合并 bean 定义:

    • 由于后置处理器可能已修改原始元数据,例如替换值中的占位符,因此清除缓存的合并 bean 定义。

这篇关于spring ioc源码-refresh();的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/558118

相关文章

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

Spring AI集成DeepSeek的详细步骤

《SpringAI集成DeepSeek的详细步骤》DeepSeek作为一款卓越的国产AI模型,越来越多的公司考虑在自己的应用中集成,对于Java应用来说,我们可以借助SpringAI集成DeepSe... 目录DeepSeek 介绍Spring AI 是什么?1、环境准备2、构建项目2.1、pom依赖2.2