Анализ процесса запуска контейнера Spring Framework Framework

Spring

Создайте среду Spring

использоватьApplicationContext context = new ClassPathXmlApplicationContext("classpath:spring.xml");способ создать контейнер Spring и просмотреть его внутренний запущенный процесс.

Весенняя версия 5.1.3.РЕЛИЗ

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.1.3.RELEASE</version>
        </dependency>

тестовый класс

public class User {
    private String username;
    private String password;

    public User(String username, String password) {
        this.username = username;
        this.password = password;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }
}

Файл конфигурации Spring

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    <bean name="user" class="com.jimisun.learnspringboot.web.User">
        <constructor-arg index="0" value="jimisun"/>
        <constructor-arg index="1" value="jimisun"/>
    </bean>

</beans>

Метод испытаний Основной

public class Main {
        public static void main(String[] args) {
            // 用我们的配置文件来启动一个 ApplicationContext
            ApplicationContext context = new ClassPathXmlApplicationContext("classpath:spring.xml");
            System.out.println("context 启动成功");
            User user = context.getBean(User.class);
            System.out.println(user.toString());
        }
}

Быстро введите Debug, чтобы просмотреть исходный код сборки контейнера IOC.

В процессе запуска ApplicationContext создается контейнер SPring Bean, затем инициализируются связанные компоненты и связанные с ними зависимости внедряются в компоненты.

Итак, нам просто нужно отладить трассировку в методе Main.ApplicationContext context = new ClassPathXmlApplicationContext("classpath:spring.xml");В этом коде показано, как Spring создает контейнер ApplicationContext и собирает информацию о конфигурации в XML-файле в контейнер.

Пошаговый анализ исходного кода Spring IOC

Шаг 1. Проверьте и настройте файл конфигурации Spring XML.

Функция: устанавливает расположение файла конфигурации для этого контекста приложения, если не задано; Spring может использовать значения по умолчанию по мере необходимости.

setConfigLocations(configLocations);

Запустите точку останова в методе Main Debug, нажмите F7 и введите его метод для просмотра, он войдетClassPathXmlApplicationContextв конструкторе класса

public class ClassPathXmlApplicationContext extends AbstractXmlApplicationContext {
  private Resource[] configResources;
  // 如果已经有 ApplicationContext 并需要配置成父子关系,那么调用这个构造方法
  public ClassPathXmlApplicationContext(ApplicationContext parent) {
    super(parent);
  }
  ...
  public ClassPathXmlApplicationContext(String[] configLocations, boolean refresh, ApplicationContext parent)
      throws BeansException {

    super(parent);
    // 根据提供的路径,处理成配置文件数组(以分号、逗号、空格、tab、换行符分割)
    setConfigLocations(configLocations);
    if (refresh) {
      refresh(); // 核心方法  剩余的所有步骤都在此方法中!!!
    }
  }
    ...
}

сначала выполнить"设置配置位置setConfigLocations"метод, разборSpringXML配置文件адрес хранится вconfigLocationsв свойствах.

	public void setConfigLocations(@Nullable String... locations) {
        //判断配置路径是否为null
		if (locations != null) {
			Assert.noNullElements(locations, "Config locations must not be null");
            //循环将配置文件路径存储到属性configLocations中
			this.configLocations = new String[locations.length];
			for (int i = 0; i < locations.length; i++) {
				this.configLocations[i] = resolvePath(locations[i]).trim();
			}
		}
		else {
			this.configLocations = null;
		}
	}

Шаг 2. Выполните подготовительные работы перед созданием контейнера компонентов.

Примечание. За исключением первого шага установки пути к файлу конфигурации XML, остальные шаги выполняются в методе refresh(); этого класса, поэтому нам нужно отладить и ввести этот метод.

Метод refresh(); выглядит следующим образом, поскольку в этом методе находится конструкция всего SpringApplication, поэтому я покажу его здесь и ознакомлюсь со всеми.

public void refresh() throws BeansException, IllegalStateException {
    &emsp;//对下面的代码块添加同步锁
   synchronized (this.startupShutdownMonitor) {
   &emsp;//第二步: 执行创建容器前的准备工作 :记录下容器的启动时间、标记“已启动”状态、处理配置文件中的占位符
      prepareRefresh();
&emsp;&emsp;//第三步:创建Bean容器,加载XML配置信息 : 如果存在容器进行销毁旧容器,创建新容器,解析XML配置文件为一个个BeanDefinition定义注册到新容器(BeanFactory)中,注意Bean未初始化 
      ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
      //第四步: 设置 BeanFactory 的类加载器,添加几个 BeanPostProcessor,手动注册几个特殊的 bean
      prepareBeanFactory(beanFactory);
      try {
         //第五步:加载并执行后置处理器 
         postProcessBeanFactory(beanFactory);
         //执行postProcessBeanFactory()方法
         invokeBeanFactoryPostProcessors(beanFactory);
          // 实例化拦截Bean创建的后置处理器beanPostProcessors
         registerBeanPostProcessors(beanFactory);
         //第六步: 初始化Spring容器的消息源
         initMessageSource();
         //第七步:初始化Spring容器事件广播器
         initApplicationEventMulticaster();
         // 空方法
         onRefresh();
         //第八步:注册事件监听器&emsp;
         registerListeners();
         //第九步核心方法:初始化(构造)所有在XML文件中配置的单例非延迟加载的bean
         finishBeanFactoryInitialization(beanFactory);
         //第十步:清理缓存,如果容器中存Bean名为lifecycleProcessor的Bean 对其进行注册,如果不存在创建一个DefaultLifecycleProcessor进行注册
         finishRefresh();
      }
      catch (BeansException ex) {
         if (logger.isWarnEnabled()) {
            logger.warn("Exception encountered during context initialization - " +
                  "cancelling refresh attempt: " + ex);
         }
         // 摧毁已经创建的单身人士以避免悬空资源。
         destroyBeans();
         // 重置'有效'标志。
         cancelRefresh(ex);
         // 向调用者传播异常。
         throw ex;
      }

      finally {
         //重置Spring核心的工具类的缓存
         resetCommonCaches();
      }
   }
}

Основная работа второго шага: подготовка, запись времени запуска контейнера, отметка состояния «запущено», обработка плейсхолдеров в конфигурационном файле, инициализация свойств события.

prepareRefresh();

protected void prepareRefresh() {
   // 记录启动时间,
   // 将 active 属性设置为 true,closed 属性设置为 false,它们都是 AtomicBoolean类型
   this.startupDate = System.currentTimeMillis();
   this.closed.set(false);
   this.active.set(true);
	//打印Logger
   	if (logger.isDebugEnabled()) {
			if (logger.isTraceEnabled()) {
				logger.trace("Refreshing " + this);
			}
			else {
				logger.debug("Refreshing " + getDisplayName());
			}
	}
	
   // 在上下文环境中初始化任何占位符属性源 空方法  默认情况下不执行任何操作。
   initPropertySources();
   // 校验 xml 配置文件
   getEnvironment().validateRequiredProperties();
   this.earlyApplicationEvents = new LinkedHashSet<ApplicationEvent>();
}

Шаг 3: Создайте контейнер Bean, загрузите и зарегистрируйте bean-компоненты

Основная работа: Уничтожение старых контейнеров, создание новых контейнеров, загрузка BeanDefinition в BeanFactory.

ConfigurableListableBeanFactory beanFactory =getFreshBeanFactory(); ​

@Override
	protected final void refreshBeanFactory() throws BeansException {
	      // 如果ApplicationContext中已经的BeanFactory属性已经有值,销毁此BeanFactory所有 Bean,关闭 BeanFactory,重新创建一个新的Bean容器设置给ApplicationContext的beanFactory属性
		if (hasBeanFactory()) {
		    //销毁容器
			destroyBeans();
			//创建类型为DefaultListableBeanFactory新容器放入BeanFactory变量中
			closeBeanFactory();
		}
		try {
		    //创建类型为DefaultListableBeanFactory新容器放入BeanFactory变量中
			DefaultListableBeanFactory beanFactory = createBeanFactory();
			//设置BeanFactory的序列化ID也就是其类名
			beanFactory.setSerializationId(getId());
			// 设置 BeanFactory 的两个配置属性:是否允许 Bean 覆盖、是否允许循环引用
			customizeBeanFactory(beanFactory);
			//这个方法将根据配置,加载各个Bean,然后放到 BeanFactory 中&emsp;注意:这里的加载并不是初始化这个Bean  而是以Key-value的形式存储在beanFactory; beanName-> beanDefinition 的 map
			loadBeanDefinitions(beanFactory);
			synchronized (this.beanFactoryMonitor) {
				this.beanFactory = beanFactory;
			}
		}
		catch (IOException ex) {
			throw new ApplicationContextException("I/O error parsing bean definition source for " + getDisplayName(), ex);
		}
	}

Шаг 4: Настройте контейнер Bean: prepareBeanFactory

Основная работа: после создания контейнера бинов некоторые специальные бины будут зарегистрированы "вручную". Официальный сайт объясняет это так: «Настройте стандартные контекстные функции фабрики, такие как ClassLoader контекста и постпроцессор».

Конкретный метод: prepareBeanFactory(фабрика);

protected void prepareBeanFactory(ConfigurableListableBeanFactory beanFactory) {
   		// 这里设置为加载当前 ApplicationContext 类的类加载器
		beanFactory.setBeanClassLoader(getClassLoader());
		// 设置 Bean的表达式解析器
		beanFactory.setBeanExpressionResolver(new StandardBeanExpressionResolver(beanFactory.getBeanClassLoader()));
		beanFactory.addPropertyEditorRegistrar(new ResourceEditorRegistrar(this, getEnvironment()));
    //默认添加一个ApplicationContextAwareProcessor的BeanPostProcessor,实现了ApplicationContextAware接口的Bean,Spring会将上下文ApplicationContext注入Bean属性中
		beanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this));
		   // 下面几行的意思就是,如果某个 bean 依赖于以下几个接口的实现类,在自动装配的时候忽略它们,
beanFactory.ignoreDependencyInterface(EnvironmentAware.class);
		beanFactory.ignoreDependencyInterface(EmbeddedValueResolverAware.class);	beanFactory.ignoreDependencyInterface(ResourceLoaderAware.class);	beanFactory.ignoreDependencyInterface(ApplicationEventPublisherAware.class);
beanFactory.ignoreDependencyInterface(MessageSourceAware.class);
beanFactory.ignoreDependencyInterface(ApplicationContextAware.class);
 /**
    * 下面几行就是为特殊的几个 bean 赋值,如果有 bean 依赖了以下几个,会注入这边相应的值,
    * 之前我们说过,"当前 ApplicationContext 持有一个 BeanFactory",这里解释了第一行
    * ApplicationContext 还继承了 ResourceLoader、ApplicationEventPublisher、MessageSource
    * 所以对于这几个依赖,可以赋值为 this,注意 this 是一个 ApplicationContext
    * 那这里怎么没看到为 MessageSource 赋值呢?那是因为 MessageSource 被注册成为了一个普通的 bean
    */	beanFactory.registerResolvableDependency(BeanFactory.class, beanFactory);	beanFactory.registerResolvableDependency(ResourceLoader.class, this);
beanFactory.registerResolvableDependency(ApplicationEventPublisher.class, this);
beanFactory.registerResolvableDependency(ApplicationContext.class, this);
		//注册早期后处理器以检测内部bean作为ApplicationListeners
		beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(this));
	// 如果检测到LoadTimeWeaver 准备编织  不是我们本章的重点无需关注
	if (beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) {
		beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory));
		beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader()));
	}
	//默认注册 environment systemEnvironment systemProperties的Bean 我们可以选择覆盖
	if (!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());
	}
}
```
	

Шаг 5: Постпроцессоры, которые обрабатывают пользовательские bean-компоненты

Основная функция: создание экземпляра Bean-компонента, реализующего интерфейсы BeanFactoryPostProcessor и BeanPostProcessors в конфигурации XML, и выполнение его метода обратного вызова. Примечание. В настоящее время обычные компоненты еще не инициализированы.

//实例化并调用XML配置中实现了BeanFactoryPostProcessors接口的的回调
postProcessBeanFactory(beanFactory);
invokeBeanFactoryPostProcessors(beanFactory);
//实例化并调用XML配置中实现了BeanPostProcessors接口的的回调
registerBeanPostProcessors(beanFactory);
//注解:这里在创建完成Bean容器后执行BeanFactoryPostProcessors接口的回调,我们可以在Bean容器初始化完成的时候完成我们自己的业务逻辑(很少用),然后是registerBeanPostProcessors(beanFactory)方法,此方法的官方解释是:"Register bean processors that intercept bean creation(如果存在则注册拦截bean创建的bean后置处理器)"

Шаг 6: Инициализируйте источник сообщений контейнера Spring

Основная функция: Инициализировать MessageSource. Если не определено в этом контексте, используется родитель.

// 初始化ApplicationContext的消息源。
initMessageSource();
	protected void initMessageSource() {
		ConfigurableListableBeanFactory beanFactory = getBeanFactory();
        //判断beanFactory中是否有messageSource的Bean
		if (beanFactory.containsLocalBean(MESSAGE_SOURCE_BEAN_NAME)) {
			this.messageSource = beanFactory.getBean(MESSAGE_SOURCE_BEAN_NAME, MessageSource.class);
			// 使MessageSource知道父MessageSource
			if (this.parent != null && this.messageSource instanceof HierarchicalMessageSource) {
				HierarchicalMessageSource hms = (HierarchicalMessageSource) this.messageSource;
				if (hms.getParentMessageSource() == null) {
				//如果没有父MessageSource,则此消息源设置为父MessageSource
					hms.setParentMessageSource(getInternalParentMessageSource());
				}
			}
			if (logger.isTraceEnabled()) {
				logger.trace("Using MessageSource [" + this.messageSource + "]");
			}
		}
		else {
			// 如果没有则创建一个默认的DelegatingMessageSource消息源
			DelegatingMessageSource dms = new DelegatingMessageSource();
			dms.setParentMessageSource(getInternalParentMessageSource());
			this.messageSource = dms;
			beanFactory.registerSingleton(MESSAGE_SOURCE_BEAN_NAME, this.messageSource);
			if (logger.isTraceEnabled()) {
				logger.trace("No '" + MESSAGE_SOURCE_BEAN_NAME + "' bean, using [" + this.messageSource + "]");
			}
		}
	}

Шаг 7. Инициализируйте средство вещания событий контейнера Spring

PS: В реальных проектах мы редко используем вещатель событий Spring, потому что теперь он распределен в приложениях и локальная связь используется редко Отличное объяснение событий о контейнере Springnuggets.capable/post/684490…

Основная функция: регистрация вещателя событий Spring для трансляции встроенных и пользовательских событий Spring.

initApplicationEventMulticaster();

	
protected void initApplicationEventMulticaster() {
	//初始化ApplicationEventMulticaster
		ConfigurableListableBeanFactory beanFactory = getBeanFactory();
		if (beanFactory.containsLocalBean(APPLICATION_EVENT_MULTICASTER_BEAN_NAME)) {
			this.applicationEventMulticaster =
					beanFactory.getBean(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, ApplicationEventMulticaster.class);
			if (logger.isTraceEnabled()) {
				logger.trace("Using ApplicationEventMulticaster [" + this.applicationEventMulticaster + "]");
			}
		}
    //如果在上下文中没有定义,则创建一个默认的SimpleApplicationEventMulticaster。
		else {
			this.applicationEventMulticaster = new SimpleApplicationEventMulticaster(beanFactory);
			beanFactory.registerSingleton(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, this.applicationEventMulticaster);
			if (logger.isTraceEnabled()) {
				logger.trace("No '" + APPLICATION_EVENT_MULTICASTER_BEAN_NAME + "' bean, using " +
						"[" + this.applicationEventMulticaster.getClass().getSimpleName() + "]");
			}
		}
	}

Шаг 8: Зарегистрируйте прослушиватель событий

Основная функция: создание экземпляра bean-компонента, реализующего интерфейс ApplicationListener.

// 注册监听器
finishBeanFactoryInitialization(beanFactory);
	protected void registerListeners() {
		//首先注册静态指定的侦听器
		for (ApplicationListener<?> listener : getApplicationListeners()) {
			getApplicationEventMulticaster().addApplicationListener(listener);
		}
        //下面是我们自定义的监听器,Spring文档中给出的建议是 "不要在这里初始化FactoryBeans:我们需要保留所有常规bean"
String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false);
		for (String listenerBeanName : listenerBeanNames) {
			getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName);
		}
        //使用已经注册的事件广播器,发布早期的应用程序事件......
		Set<ApplicationEvent> earlyEventsToProcess = this.earlyApplicationEvents;
		this.earlyApplicationEvents = null;
		if (earlyEventsToProcess != null) {
			for (ApplicationEvent earlyEvent : earlyEventsToProcess) {
				getApplicationEventMulticaster().multicastEvent(earlyEvent);
			}
		}
	}

Шаг 9: Создание экземпляров всех одноэлементных компонентов

Функция: на этом этапе также регистрируются специальные bean-компоненты в файле конфигурации Spring.xml, и те, которые должны быть вызваны, также вызываются, оставляя обычные bean-компоненты, экземпляры которых создаются на этом шаге. (Это только одноэлементный Бин, который не создается с задержкой), то есть инициализация фабрики Бин (ApplicationContext) была завершена на этом шаге.

// 实例化所有SPring.xml配置文件中配置的非延迟实例化的单例Bean
finishBeanFactoryInitialization(beanFactory);
protected void finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory) {
		// 初始化此上下文的转换服务
		if (beanFactory.containsBean(CONVERSION_SERVICE_BEAN_NAME) &&
				beanFactory.isTypeMatch(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class)) {
			beanFactory.setConversionService(
					beanFactory.getBean(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class));
		}

		//如果没有bean后处理器,则注册默认的嵌入值解析器(例如PropertyPlaceholderConfigurer bean)之前注册过;此时,主要用于注释属性值的分辨率。
		if (!beanFactory.hasEmbeddedValueResolver()) {
			beanFactory.addEmbeddedValueResolver(strVal -> getEnvironment().resolvePlaceholders(strVal));
		}

		// 尽早初始化LoadTimeWeaverAware bean以允许尽早注册其变换器。
		String[] weaverAwareNames = beanFactory.getBeanNamesForType(LoadTimeWeaverAware.class, false, false);
		for (String weaverAwareName : weaverAwareNames) {
			getBean(weaverAwareName);
		}
		// 停止使用临时ClassLoader进行类型匹配。
		beanFactory.setTempClassLoader(null);
		// 允许缓存所有bean定义元数据,而不期望进一步的更改。
		beanFactory.freezeConfiguration();
		// 实例化所有剩余(非延迟初始化)单例。
		beanFactory.preInstantiateSingletons();
	}

Шаг 10. Завершите инициализацию контейнера ApplicationContext.

Функция: выполнение операций при создании связанных контейнеров и повторное использование связанных ресурсов.

 finishRefresh();
 resetCommonCaches();
	protected void finishRefresh() {
		//清除上下文级资源缓存(例如来自扫描的ASM元数据)。
		clearResourceCaches();
		//为此上下文初始化生命周期处理器。
		initLifecycleProcessor();
		// 首先将刷新传播到生命周期处理器。
		getLifecycleProcessor().onRefresh();
		// 广播最终事件
		publishEvent(new ContextRefreshedEvent(this));
		// 如果处于活动状态,请参与LiveBeansView 
		LiveBeansView.registerApplicationContext(this);
	}
//清除一些单例的工具类的缓存
protected void resetCommonCaches() {
		ReflectionUtils.clearCache();
		AnnotationUtils.clearCache();
		ResolvableType.clearCache();
CachedIntrospectionResults.clearClassLoader(getClassLoader());
	}

знания, которые нужно изучить

  • Является ли Bean управляемым в контейнере Spring классом?

Вы можете видеть, что отношение сопоставления определения Bean-компонента в хранилищах контейнеров Bean-компонентовkey(String)->GenericBeanDefinitionотображение, затемGenericBeanDefinitionЧто это такое?

BeanDefinitionспас нашBean 信息, например, на какой класс указывает bean-компонент, является ли он одноэлементным, загружается ли он с отложенной загрузкой, от каких bean-компонентов зависит bean-компонент и так далее.

  • Управляется ли SpringBeans ApplicationContext?

пройти черезDebugВ процессе мы видим, что мы используемClassPathXmlApplicationContextпостроенApplicationContextОбъект фактически поддерживает свойство с внутренним именемbeanFactory, все наши SpringBeans определены в этом свойстве, то естьbeanFactoryЭто свойство является контейнером,ApplicationContextПросто сделайте слой упаковки.beanFactoryЧто это такое?

public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory
		implements ConfigurableListableBeanFactory, BeanDefinitionRegistry, Serializable {
		...
}

3

можно увидетьDefaultListableBeanFactoryКласс также является контейнером Bean и наследует функции всех других контейнеров.Можно сказать, что это самый мощный контейнер, например, он имеет (расслоение, получение нескольких контейнеров, функции внедрения....)

краткое содержание главы

В первый раз, когда я читал исходный код, я был более осторожен. Из-за моей физической болезни и ошибок это было через несколько дней после выхода этой главы. В целом, в этой главе нет важных моментов.loadBeanDefinitions(beanFactory)Spring如何将XMl文件的配置装载入Bean工厂, и каждый из следующих комментариев может открыть новую длинную статью, а позже, насколько это возможно, вКолонка углубленного анализа Spring FrameworkИзучите исходный код общей архитектуры Spring более подробно в этой статье.Эта статья основана на примечаниях к изучению оригинального https://juejin.cn/post/6844903694039793672, чтобы более четко показать шаги.

Этот учебник относится к колонке углубленного анализа Spring Framework инженеров Java и соответствующему каталогу сообщений в блогах этой серии.Колонка глубокого анализа Spring Framework для инженеров Java