1、AbstractApplicationContext的prepareRefresh()方法调用。

2、XmlBeanDefinitionReader的loadBeanDifinitions(BeanDefinitionRegistry)方法调用。加载spring配置文件,加载完成之后回调BeanDefinitionRegistry的preInstantiateSingletons()方法,这个方法由DefaultListableBeanFactory类实现。

3、AbstractApplicationContext的obtainFreshBeanFactory()方法调用。

4、DefaultListableBeanFactory的preInstantiateSingletons()方法调用。为所有xml文件中定义(以及采用标注方式定义)的bean建立实例。

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

3、注册Bean后处理器:根据反射机制从BeanDefinitionRegistry中找出所有BeanPostProcessor类型的Bean,并将它们注册到容器的BeanPostProcessor的注册表中;

AbstractApplicationContext中的代码如下:

protected void registerBeanPostProcessors(ConfigurableListableBeanFactory beanFactory) {

  String[] processorNames = beanFactory.getBeanNamesForType(BeanPostProcessor.class, true, false);

  

}

4、初始化消息源:初始化容器的国际化信息资源;

AbstractApplicationContext中的代码如下:

protected void initMessageSource() {

  //补充

}

5、初始化应用上下文事件广播器;(观察者模式中的具体主题角色,持有观察者角色的集合,称为注册表)。

AbstractApplicationContext拥有一个applicationEventMulticaster成员变量,applicationEventMulticaster提供了容器监听器的注册表,称其为事件广播器。在第七步中将会将事件监听器装入其中。

AbstractApplicationContext中的代码如下:

private ApplicationEventMulticaster applicationEventMulticaster;

protected void initApplicationEventMulticaster() {

  //"applicaitonEventMulticaster",先看配置文件中有无配置该类型。

  if (containsLocalBean(APPLICATION_EVENT_MULTICASTER_BEAN_NAME))  {

    this.applicationEventMulticaster = (ApplicationEventMulticaster) getBean(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, ApplicationEventMulticaster.class);

  } else {

    //若没有,则应用Spring框架提供的事件广播器实例

    this.applicationEventMulticaster = new SimpleApplicationEventMulticaster();

  }

}

public boolean containsLocalBean(String name) {

  return getBeanFactory().containsLocalBean(name);

}

public Object getBean(String name, Class requiredType) throws BeansException {

  return getBeanFactory().getBean(name, requiredType);

}

Spring初始化事件广播器,用户可以在配置文件中为容器定义一个自定义的事件广播器(bean的名称要为"applicaitonEventMulticaster"),只要实现ApplicationEventMulticaster就可以了,Spring在此会根据beanfactory自动获取。如果没有找到外部配置的事件广播器,Spring使用SimpleApplicationEventMulticaster作为事件广播器。

6、初始化其他特殊Bean:这是一个钩子方法,子类可以借助这个钩子方法执行一些特殊的操作:如AbstractRefreshableWebApplicationContext就使用该钩子方法执行初始化ThemeSource操作;

protected void onRefresh() throws BeansException {

  

}

7、注册事件监听器(观察者模式中的观察者角色)

Spring根据上下文持有的beanfactory对象(默认是DefaultListableBeanFactory对象),从它的BeanDefinitionRegistry中找出所有实现ApplicationListener的bean,将BeanDefinition对象生成bean,注册为容器的事件监听器,实际的操作就是将其添加到事件广播器所提供的监听器注册表中。

AbstractApplicationContext中的代码如下:

private List<ApplicationListener> applicationListeners = new ArrayList<ApplicationListener>();

public List<ApplicationListener> getApplicationListeners() {

  return this.applicationListeners;

}

protected void registerListeners() {

  for (Iterator<ApplicationListener> it=getApplicationListeners().iterator(); it.hasNext();) {

    addListener((ApplicationListener) it.next());

  }

  //获取ApplicationListener类型的所有bean,即事件监听器

  Collection<ApplicationListener> listenerBeans = getBeansOfType(ApplicationListener.class, true, false).values;

  for (Iterator<ApplicationListener> it=listenerBeans.iterator(); it.hasNext();) {

    addListener((ApplicationListener)it.next());

  }

}

public Map getBeansOfType(Class type, boolean includePrototypes, boolean allowEageInit) throws BeansException {

  return getBeanFactory().getBeansOfType(type, includePrototypes, allowEagerInit);

}

protected void addListener(ApplicationListener listener) {

  getApplicationEventMulticaster().addApplicationListener(listener);

}

8、初始化singleton的Bean:实例化所有singleton的Bean,并将它们放入Spring容器的缓存中(DefaultListableBeanFactory,它实现了BeanDefinitionRegistry),这就是和直接在应用中使用BeanFactory的区别之处,在创建ApplicationContext对象时,不仅创建了一个BeanFactory对象,并且还应用它实例化所有单实例的bean。

AbstractApplicationContext中的代码如下:

beanFactory.preInstantiateSingletons();

spring容器启动过程的更多相关文章

  1. spring容器启动过程理解

    一.一切从手动启动IoC容器开始 ClassPathResource resource = new ClassPathResource("bean.xml"); DefaultLi ...

  2. 转:spring的启动过程-spring和springMVC父子容器的原理

    要想很好理解这三个上下文的关系,需要先熟悉spring是怎样在web容器中启动起来的.spring的启动过程其实就是其IoC容器的启动过程,对于web程序,IoC容器启动过程即是建立上下文的过程. s ...

  3. Spring源码解析-Web容器启动过程

    Web容器启动过程,主要讲解Servlet和Spring容器结合的内容. 流程图如下: Web容器启动的Root Context是有ContextLoaderListener,一般使用spring,都 ...

  4. spring的启动过程就是创建ioc容器的过程

    1. spring简介 spring的最基本的功能就是创建对象及管理这些对象之间的依赖关系,实现低耦合.高内聚.还提供像通用日志记录.性能统计.安全控制.异常处理等面向切面的能力,还能帮我们管理最头疼 ...

  5. Spring源码分析专题 —— IOC容器启动过程(上篇)

    声明 1.建议先阅读<Spring源码分析专题 -- 阅读指引> 2.强烈建议阅读过程中要参照调用过程图,每篇都有其对应的调用过程图 3.写文不易,转载请标明出处 前言 关于 IOC 容器 ...

  6. Spring Boot启动过程(四):Spring Boot内嵌Tomcat启动

    之前在Spring Boot启动过程(二)提到过createEmbeddedServletContainer创建了内嵌的Servlet容器,我用的是默认的Tomcat. private void cr ...

  7. Spring MVC启动过程(1):ContextLoaderListener初始化

    此文来自https://my.oschina.net/pkpk1234/blog/61971 (写的特别好)故引来借鉴 Spring MVC启动过程 以Tomcat为例,想在Web容器中使用Spirn ...

  8. 【Spring】简述@Configuration配置类注册BeanDefinition到Spring容器的过程

    概述 本文以SpringBoot应用为基础,尝试分析基于注解@Configuration的配置类是如何向Spring容器注册BeanDefinition的过程 其中主要分析了 Configuratio ...

  9. Spring Boot启动过程(七):Connector初始化

    Connector实例的创建已经在Spring Boot启动过程(四):Spring Boot内嵌Tomcat启动中提到了: Connector是LifecycleMBeanBase的子类,先是设置L ...

随机推荐

  1. fread与fwrite的自我理解

    size_t  fread(void* buff,size_t size,size_t count,FILE* stream) 参数1:读取到该buff所指向的内存空间中 参数2:每次读取的字节数,单 ...

  2. URAL 1139 City Blocks(数论)

    The blocks in the city of Fishburg are of square form. N avenues running south to north and Mstreets ...

  3. start.s 解析(一)

    可以参考 : http://blog.csdn.net/bluesummerg/article/details/5940452 (强大的反汇编) http://www.cnblogs.com/yanh ...

  4. Linux内核之旅 List_entry()

    #include "iostream" #define List_entry(type,member)\ (type *)(()->data)) ) using namesp ...

  5. ppaer 67 : matlab 函数errorbar

    MATLAB ERRORBAR 这个函数的意思是:ERRORBAR(X,Y,L,U),X是自变量,Y是因变量,L是Y的变动下限,U是Y的变动上限 errorbar(X,Y,E)  X是自变量,Y是因变 ...

  6. mysql设置时区方法

    set global time_zone = '+2:00'; ##修改mysql全局时区 set time_zone = '+2:00'; ##修改当前会话时区 flush privileges; ...

  7. RMAN备份演练进阶篇

    前篇介绍了通过rman进行各种备份,进阶篇则主要是rman的一些功能扩展和增加功能,利用前篇你已经完全可以完成数据库的备份,而通过本篇你可以更好更方便的完成数据库的备份. 一.建立增量备份 如果数据库 ...

  8. weka 文本分类(1)

    一.初始化设置 1 jvm out of memory 解决方案: 在weka SimpleCLI窗口依次输入java -Xmx 1024m 2 修改配置文件,使其支持中文: 配置文件是在Weka安装 ...

  9. 如何写一个c++插件化系统

    1.为什么需要插件化系统 “编程就是构建一个一个自己的小积木, 然后用自己的小积木搭建大系统”. 但是程序还是会比积木要复杂, 我们的系统必须要保证小积木能搭建出大的系统(必须能被组合),有必须能使各 ...

  10. 继承(引用~析构~virtual)

    [1]程序1 #include <iostream> using namespace std; class Base { private: int m_nBase; public: Bas ...