spring framework 4 源码阅读

前面写了几篇spring 的介绍文章,感觉与主题不是很切合。重新整理下思路,从更容易理解的角度来写下文章。

spring 的骨架

spring 的骨架,也是spring 的核心包。主要包含三个内容

1.context:spring 的上线文-------导演

2.core:spring的核心包,主要包括spring所以用到的工具-------道具

3.beans:spring的bean实例 -------演员

导演负责安排演出,演员负责按照导演的指示来演出,演出过程中需要使用道具。

我想大家看完这些图片之后就明白大致的包关系了。

spring包结构

大家看到相应包内容。

core包侧重于帮助类,操作工具,beans包更侧重于bean实例的描述。context更侧重全局控制,功能衍生。

下面我们就针对context和factory对类的关系继续一个基本概括:

核心类之间的关系

我们先来看下bean包下的beanfactory类,以及抽象类等。

可以看到在接口的实现泛化的过程中,每一个接口在继承父接口的同时,也继承了父接口的一些方法。这就可以看出面向接口变成微妙之处。

BeanFactory【所有BeanFactory的父类】

可以看到beanfactory中定义了一些基本方法,包括根据名称获取bean实例等。

HierarchicalBeanFactory【层次化的BeanFactory】

可以看到此接口实现了层次化,及获取beanFactory的父容器

LisableBeanFactory列表式Beanfactory

可以看到为beanfactory设置了列表的功能,并且规划了如何从列表中取出相应的方法的能力。

小结:

从上述类命名以及接口规划可以看到,通过接口的不断继承,beanfactory被不断的丰富抽象起来。层层细分之后,没有个类都的职责都变的单一了,同时在扩展该的属性时也变的更加方便。针对源代码,最好的办法还是根据名称来,最方便。

context【上下线文】

可以看到到了context的初始化不同于beanfactory,可以侧重于抽象类型,具体的方法实现。

里面大部分方法使用了模板方法的设计模式,父类调用抽象方法,抽象方法在子类中实现,对象的独立性。

主要分成三种context:XML,Annotation,Groovy针对三种形式。

registry【实例或者bean描述注册器】

将初始化完成的bean注册的容器中,针对单例来部分,缓存单例实例。针对beanDefinition部分,缓存bean描述。

Strategy【初始化策略】

两种初始化策略 一种是简单策略,一种是cgilib的策略,当时这里使用的模式是策略模式。

context的初始化

	/**
	 * 在parent下创建ClassPathXmlApplicaitonContext,
	 * 从XML中读取所有Bean定义.
	 * @param configLocations 配置文件路径如c:\simpleContext.xml
	 * @param refresh 是否需要自动刷新context,refresh-->重新加载
	 * 加载所有的bean定义,创建所有单例.
	 * refresh为true的时候, 根据context来手工刷新
	 * @param parent the parent context
	 * @throws BeansException if context creation failed
	 * @see #refresh()
	 */
	public ClassPathXmlApplicationContext(String[] configLocations, boolean refresh, ApplicationContext parent)
			throws BeansException {
		//初始化XmlApplicationContext
		super(parent);
		//转换配置文件的路径
		setConfigLocations(configLocations);
		if (refresh) {
			//重新刷新原有的context,这一篇的重点
			refresh();
		}
	}

下面我们来看下AbstractApplicationContext.refresh()方法

	//加载或刷新持久的配置,可能是xml文件,properties文件,或者关系型数据库的概要。
	//做为一个启动方法,如果初始化失败将会销毁已经创建好的单例,避免重复加载配置文件。
	//换句话说,在执行这个方法之后,要不全部加载单例,要不都不加载
	public void refresh() throws BeansException, IllegalStateException
	{
		synchronized (this.startupShutdownMonitor)
		{
			// 初始化配置准备刷新,验证环境变量中的一些必选参数
			prepareRefresh();

			// 告诉继承类销毁内部的factory创建新的factory的实例
			// 初始化Bean实例
			ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();

			// 初始化beanFactroy的基本信息,包括classloader,environment,忽略的注解等
			prepareBeanFactory(beanFactory);

			try {
				// beanfactory内部的postProcess,可以理解为context中PostProcess的补充
				beanFactory.postProcessBeanFactory(beanFactory);

				// 执行BeanFactoryPostProcessor(在beanFactory初始化过程中,bean初始化之前,修改beanfactory参数)、
				// BeanDefinitionRegistryPostProcessor 其实也是继承自BeanFactoryPostProcessor,
				// 多了对BeanDefinitionRegistry的支持invokeBeanFactoryPostProcessors(beanFactory);
				// 执行postProcess,那BeanPostProcessor是什么呢,是为了在bean加载过程中修改bean的内容,
				// 使用分的有两个而方法Before、After分别对应初始化前和初始化后
				registerBeanPostProcessors(beanFactory);

				// 初始化MessageSource,主要用作I18N本地化的内容
				initMessageSource();

				// 初始化事件广播ApplicationEventMulticaster,使用观察者模式,对注册的ApplicationEvent时间进行捕捉
				initApplicationEventMulticaster();

				// 初始化特殊bean的方法
				onRefresh();

				// 将所有ApplicationEventListener注册到ApplicationEventMulticaster中
				registerListeners();

				// 初始化所有不为lazy-init的bean,singleton实例
				finishBeanFactoryInitialization(beanFactory);

				// 初始化lifecycle的bean并启动(例如quartz的定时器等),如果开启JMX则将ApplicationContext注册到上面
				finishRefresh();
			}
			catch (BeansException ex)
			{
				//销毁已经创建单例
				resources.destroyBeans();

				// 将context的状态转换为无效,标示初始化失败
				flag.cancelRefresh(ex);

				// 将异常传播到调用者
				throw ex;
			}
		}
	}

我们从时序图来看启动上述初始化(门面模式facade)

时间: 2024-10-11 06:45:55

spring framework 4 源码阅读的相关文章

spring framework 4 源码阅读 --- 前期准备

在开始看代码之前,需要做的第一件事是下载代码. 在这里:https://github.com/spring-projects/spring-framework 下载完成了发现使用gradle做的源代码的构建管理工具的. 值得庆幸的事情,spring为导入eclipse写了脚本工具,包解压后结构如下: 发现了一个脚本文件:import-into-eclipse.bat,今天负责导入的就是他了, 等待漫长的编译,这里有个问题,在没有安装gradle的时候,脚本会自动下载gradle 执行完成后是这样

spring framework 4 源码阅读(2)---从ClassPathXmlApplicationContext开始

Application初始化日志 15:23:12.790 [main] DEBUG o.s.core.env.StandardEnvironment - Adding [systemProperties] PropertySource with lowest search precedence 15:23:12.797 [main] DEBUG o.s.core.env.StandardEnvironment - Adding [systemEnvironment] PropertySourc

spring framework项目源码github托管地址

方法一:直接下载,github托管地址:http://repo.spring.io/simple/libs-release-local/org/springframework/spring/ 方法二:svn检出 一次性源码全量包下载的方法.spring的项目包会托管到github网站上,所以可以从github网站上得到spring任意版本的全量包(使用svn客户端下载),方法如下:1.安装svn客户端,随便使用自己喜欢的一个就行,通常用的最多的是TortoiseSVN.自己下载安装.2.检出(下

Spring依赖注入源码阅读笔记

一.调用栈 一次典型的Spring依赖注入的调用栈: DefaultListableBeanFactory:getBean() AbstractBeanFactory:doGetBean() AbstractAutowireCapableBeanFactory:createBean() AbstractAutowireCapableBeanFactory:createBeanInstance() SimpleInstantiationStategy:instantiate() AbstractA

Spring源码阅读:Spring MVC 如何处理HTTP请求

Spring MVC 对HTTP请求的处理流程 通过之前的源码阅读,知道了ApplicationContext初始的过程,也知道了Spring MVC环境的初始化过程,今天就来了解一下SpringMVC是如何处理HTTP请求的. HTTP请求根据请求方式可以分为GET.POST.PUT.DELETE.OPTIONS.TRACE,最常用的还是GET和POST. Spring对于这几种HTTP请求的处理都是使用了processRequest(req,rep); @Override protected

Spring源码阅读:IOC容器的设计与实现(二)——ApplicationContext

上一主题中,了解了IOC容器的基本概念,以及BeanFactory的设计与实现方式,这里就来了解一下ApplicationContext方式的实现. ApplicationContext 在Spring的参考文档中,为啥要推荐使用ApplicationContext?它能给我们的应用带来什么好处呢?作为BeanFactory的实现之一,它又是如何设计的?在SpringMVC中使用的WebApplictionContext\XmlApplicationContext与之有何关联? Applicat

Spring源码阅读:Spring声明式事务处理和编程式事务处理的设计与实现

之前的学习,了解了Spring事务管理的基础框架(查看).Spring在此基础上又提到了声明式事务管理和编程式事务管理.这里就来看看Spring是如何实现的. Spring声明式事务与EJB事务管理对比 Spring的声明式管理,类似于EJB的CMT,但又有不同.他们的不同之处有: 1)EJB的CMT是与JTA结合使用,而Spring框架的声明式事务管理可以在任何环境下工作.既可以使用全局事务管理,如JTA,也可以使用局部事务管理如JDBCJPA.Hibernate.JDO等. 2)可以在任何类

Spring源码阅读:Spring如何支持各种ORM框架

为了让开发程序更容易,到现在为止,已经有很多ORM框架了,例如:JPA,JDO,Hibernate,Mybatis(之前版本是IBatis)等等.也正因为已经有这么多优秀的ORM框架,Spring团队并没有自己开发一套ORM框架,而是对这些框架都进行了支持,让这些框架在Spring环境下可以得到完全的应用. 通常,在Spring环境下使用这些ORM框架时,都会通过一个Template来使用.Spring对这些框架的集成是这样的: 例如Hibernate,在使用Hibernate时(没有在Spri

Spring源码阅读:Spring事务管理的基础

上一节了解了全局事务与局部事务以及Spring提供的两种事务模式:编程式事务与声明式事务. 不论是编程式的事务处理,还是声明式的事务处理.他们都要对局部事务和全局事务以支持,也就是说要对JDBC进行支持.ORM框架,同时也要对JTA进行支持.他们的公共部分是commit,rollback.通过这一节的了解,我相信以后配置Spring事务时,就不需要在去网上查资料了或者去查Spring的参考文档了. 因此,Spring设计了如下的事务管理框架: 从上面的类图中和容易可以看出分为三部分:Platfo