Spring 是一个轻量级的企业级应用开发框架,于 2004 年由 Rod Johnson 发布了 1.0 版本。通过十几年的迭代,如今的 Spring 框架已经很是成熟了。Spring 包含了众多模块,包括但不限于 Core、Bean、Context、AOP 和 Web 等。在今天,咱们彻底可使用 Spring 所提供的一站式解决方案开发出咱们所须要的应用。做为 Java 程序员,咱们会常常和 Spring 框架打交道,因此仍是颇有必要弄懂 Spring 的原理。mysql
本文是 Spring IOC 容器源码分析系列文章的第一篇文章,将会着重介绍 Spring 的一些使用方法和特性,为后续的源码分析文章作铺垫。另外须要特别说明一下,本系列的源码分析文章是基于Spring 4.3.17.RELEASE
版本编写的,而非最新的5.0.6.RELEASE
版本。好了,关于简介先说到这里,继续来讲说下面的内容。git
写 Spring IOC 这一块的文章,挺让我纠结的。我本来是打算在一篇文章中分析全部的源码,可是后来发现文章实在太长。主要是由于 Spring IOC 部分的源码实在太长,将这一部分的源码贴在一篇文章中仍是很壮观的。固然估计你们也没兴趣读下去,因此决定对文章进行拆分。这里先贴一张文章切分前的目录结构:程序员
如上图,由目录能够看出,假使在一篇文章中写完全部内容,文章的长度将会很是长。因此在通过思考后,我会将文章拆分红一系列的文章,以下:github
上面文章对应的源码分析工做均已经完成,全部的文章将会在近期内进行更新。spring
Spring 是分模块开发的,Spring 包含了不少模块,其中最为核心的是 bean 容器相关模块。像 AOP、MVC、Data 等模块都要依赖 bean 容器。这里先看一下 Spring 框架的结构图:sql
图片来源:Spring 官方文档segmentfault
从上图中能够看出Core Container
处于整个框架的最底层(忽略 Test 模块),在其之上有 AOP、Data、Web 等模块。既然 Spring 容器是最核心的部分,那么你们若是要读 Spring 的源码,容器部分必须先弄懂。本篇文章做为 Spring IOC 容器的开篇文章,就来简单介绍一下容器方面的知识。请继续往下看。app
本章将会介绍 IOC 中的部分特性,这些特性均会在后面的源码分析中悉数到场。若是你们不是很熟悉这些特性,这里能够看一下。框架
alias 的中文意思是“别名”,在 Spring 中,咱们可使用 alias 标签给 bean 起个别名。好比下面的配置:ide
1 2 3 4 5 |
<bean id="hello" class="xyz.coolblog.service.Hello"> <property name="content" value="hello"/> </bean> <alias name="hello" alias="alias-hello"/> <alias name="alias-hello" alias="double-alias-hello"/> |
这里咱们给hello
这个 beanName 起了一个别名alias-hello
,而后又给别名alias-hello
起了一个别名double-alias-hello
。咱们能够经过这两个别名获取到hello
这个 bean 实例,好比下面的测试代码:
1 2 3 4 5 6 7 8 9 10 |
public class ApplicationContextTest { @Test public void testAlias() { String configLocation = "application-alias.xml"; ApplicationContext applicationContext = new ClassPathXmlApplicationContext(configLocation); System.out.println(" alias-hello -> " + applicationContext.getBean("alias-hello")); System.out.println("double-alias-hello -> " + applicationContext.getBean("double-alias-hello")); } } |
测试结果以下:
本小节,咱们来了解一下 autowire 这个特性。autowire 即自动注入的意思,经过使用 autowire 特性,咱们就不用再显示的配置 bean 之间的依赖了。把依赖的发现和注入都交给 Spring 去处理,省时又省力。autowire 几个可选项,好比 byName、byType 和 constructor 等。autowire 是一个经常使用特性,相信你们都比较熟悉了,因此本节咱们就 byName 为例,快速结束 autowire 特性的介绍。
当 bean 配置中的 autowire = byName 时,Spring 会首先经过反射获取该 bean 所依赖 bean 的名字(beanName),而后再经过调用 BeanFactory.getName(beanName) 方法便可获取对应的依赖实例。autowire = byName 原理大体就是这样,接下来咱们来演示一下。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
public class Service { private Dao mysqlDao; private Dao mongoDao; // 忽略 getter/setter @Override public String toString() { return super.toString() + "\n\t\t\t\t\t{" + "mysqlDao=" + mysqlDao + ", mongoDao=" + mongoDao + '}'; } } public interface Dao {} public class MySqlDao implements Dao {} public class MongoDao implements Dao {} |
配置以下:
1 2 3 4 5 6 7 8 9 10 11 |
<bean name="mongoDao" class="xyz.coolblog.autowire.MongoDao"/> <bean name="mysqlDao" class="xyz.coolblog.autowire.MySqlDao"/> <!-- 非自动注入,手动配置依赖 --> <bean name="service-without-autowire" class="xyz.coolblog.autowire.Service" autowire="no"> <property name="mysqlDao" ref="mysqlDao"/> <property name="mongoDao" ref="mongoDao"/> </bean> <!-- 经过设置 autowire 属性,咱们就不须要像上面那样显式配置依赖了 --> <bean name="service-with-autowire" class="xyz.coolblog.autowire.Service" autowire="byName"/> |
测试代码以下:
1 2 3 4 |
String configLocation = "application-autowire.xml"; ApplicationContext applicationContext = new ClassPathXmlApplicationContext(configLocation); System.out.println("service-without-autowire -> " + applicationContext.getBean("service-without-autowire")); System.out.println("service-with-autowire -> " + applicationContext.getBean("service-with-autowire")); |
测试结果以下:
从测试结果能够看出,两种方式配置方式都能完成解决 bean 之间的依赖问题。只不过使用 autowire 会更加省力一些,配置文件也不会冗长。这里举的例子比较简单,假使一个 bean 依赖了十几二十个 bean,再手动去配置,恐怕就很难受了。
FactoryBean?看起来是否是很像 BeanFactory 孪生兄弟。不错,他们看起来很像,可是他们是不同的。FactoryBean 是一种工厂 bean,与普通的 bean 不同,FactoryBean 是一种能够产生 bean 的 bean,好吧提及来很绕嘴。FactoryBean 是一个接口,咱们能够实现这个接口。下面演示一下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
public class HelloFactoryBean implements FactoryBean<Hello> { @Override public Hello getObject() throws Exception { Hello hello = new Hello(); hello.setContent("hello"); return hello; } @Override public Class<?> getObjectType() { return Hello.class; } @Override public boolean isSingleton() { return true; } } |
配置以下:
1 |
<bean id="helloFactory" class="xyz.coolblog.service.HelloFactoryBean"/> |
测试代码以下:
1 2 3 4 5 6 7 8 9 10 |
public class ApplicationContextTest { @Test public void testFactoryBean() { String configLocation = "application-factory-bean.xml"; ApplicationContext applicationContext = new ClassPathXmlApplicationContext(configLocation); System.out.println("helloFactory -> " + applicationContext.getBean("helloFactory")); System.out.println("&helloFactory -> " + applicationContext.getBean("&helloFactory")); } } |
测试结果以下:
由测试结果能够看到,当咱们调用 getBean(“helloFactory”) 时,ApplicationContext 会返回一个 Hello 对象,该对象是 HelloFactoryBean 的 getObject 方法所建立的。若是咱们想获取 HelloFactoryBean 自己,则能够在 helloFactory 前加上一个前缀&
,即&helloFactory
。
介绍完 FactoryBean,本节再来看看了一个和工厂相关的特性 – factory-method。factory-method 可用于标识静态工厂的工厂方法(工厂方法是静态的),直接举例说明吧:
1 2 3 4 5 6 7 8 |
public class StaticHelloFactory { public static Hello getHello() { Hello hello = new Hello(); hello.setContent("created by StaticHelloFactory"); return hello; } } |
配置以下:
1 |
<bean id="staticHelloFactory" class="xyz.coolblog.service.StaticHelloFactory" factory-method="getHello"/> |
测试代码以下:
1 2 3 4 5 6 7 8 9 |
public class ApplicationContextTest { @Test public void testFactoryMethod() { String configLocation = "application-factory-method.xml"; ApplicationContext applicationContext = new ClassPathXmlApplicationContext(configLocation); System.out.println("staticHelloFactory -> " + applicationContext.getBean("staticHelloFactory")); } } |
测试结果以下:
对于非静态工厂,须要使用 factory-bean 和 factory-method 两个属性配合。关于 factory-bean 这里就不继续说了,留给你们本身去探索吧。
lookup-method 特性可能你们用的很少(我也没用过),不过它也是个有用的特性。在介绍这个特性前,先介绍一下背景。咱们经过 BeanFactory getBean 方法获取 bean 实例时,对于 singleton 类型的 bean,BeanFactory 每次返回的都是同一个 bean。对于 prototype 类型的 bean,BeanFactory 则会返回一个新的 bean。如今考虑这样一种状况,一个 singleton 类型的 bean 中有一个 prototype 类型的成员变量。BeanFactory 在实例化 singleton 类型的 bean 时,会向其注入一个 prototype 类型的实例。可是 singleton 类型的 bean 只会实例化一次,那么它内部的 prototype 类型的成员变量也就不会再被改变。但若是咱们每次从 singleton bean 中获取这个 prototype 成员变量时,都想获取一个新的对象。这个时候怎么办?举个例子(该例子源于《Spring 揭秘》一书),咱们有一个新闻提供类(NewsProvider),这个类中有一个新闻类(News)成员变量。咱们每次调用 getNews 方法都想获取一条新的新闻。这里咱们有两种方式实现这个需求,一种方式是让 NewsProvider 类实现 ApplicationContextAware 接口(实现 BeanFactoryAware 接口也是能够的),每次调用 NewsProvider 的 getNews 方法时,都从 ApplicationContext 中获取一个新的 News 实例,返回给调用者。第二种方式就是这里的 lookup-method 了,Spring 会在运行时对 NewsProvider 进行加强,使其 getNews 能够每次都返回一个新的实例。说完了背景和解决方案,接下来就来写点测试代码验证一下。
在演示两种处理方式前,咱们先来看看不使用任何处理方式,BeanFactory 所返回的 bean 实例状况。相关类定义以下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
public class News { // 仅演示使用,News 类中无成员变量 } public class NewsProvider { private News news; public News getNews() { return news; } public void setNews(News news) { this.news = news; } } |
配置信息以下:
1 2 3 4 |
<bean id="news" class="xyz.coolblog.lookupmethod.News" scope="prototype"/> <bean id="newsProvider" class="xyz.coolblog.lookupmethod.NewsProvider"> <property name="news" ref="news"/> </bean> |
测试代码以下:
1 2 3 4 5 |
String configLocation = "application-lookup-method.xml"; ApplicationContext applicationContext = new ClassPathXmlApplicationContext(configLocation); NewsProvider newsProvider = (NewsProvider) applicationContext.getBean("newsProvider"); System.out.println(newsProvider.getNews()); System.out.println(newsProvider.getNews()); |
测试结果以下:
从测试结果中能够看出,newsProvider.getNews() 方法两次返回的结果都是同样的,这个是不知足要求的。
咱们让 NewsProvider 实现 ApplicationContextAware 接口,实现代码以下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
public class NewsProvider implements ApplicationContextAware { private ApplicationContext applicationContext; private News news; /** 每次都从 applicationContext 中获取一个新的 bean */ public News getNews() { return applicationContext.getBean("news", News.class); } public void setNews(News news) { this.news = news; } @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { this.applicationContext = applicationContext; } } |
配置和测试代码同上,测试结果以下:
这里两次获取的 news 并就不是同一个 bean 了,知足了咱们的需求。
使用 lookup-method 特性,配置文件须要改一下。以下:
1 2 3 4 |
<bean id="news" class="xyz.coolblog.lookupmethod.News" scope="prototype"/> <bean id="newsProvider" class="xyz.coolblog.lookupmethod.NewsProvider"> <lookup-method name="getNews" bean="news"/> </bean> |
NewsProvider 的代码沿用 4.5.1 小节以前贴的代码。测试代码稍微变一下,以下:
1 2 3 4 5 6 |
String configLocation = "application-lookup-method.xml"; ApplicationContext applicationContext = new ClassPathXmlApplicationContext(configLocation); NewsProvider newsProvider = (NewsProvider) applicationContext.getBean("newsProvider"); System.out.println("newsProvider -> " + newsProvider); System.out.println("news 1 -> " + newsProvider.getNews()); System.out.println("news 2 -> " + newsProvider.getNews()); |
测试结果以下:
从上面的结果能够看出,new1 和 new2 指向了不一样的对象。同时,你们注意看 newsProvider,彷佛变的很复杂。由此可看出,NewsProvider 被 CGLIB 加强了。
当一个 bean 直接依赖另外一个 bean,可使用 <ref/>
标签进行配置。不过如某个 bean 并不直接依赖于其余 bean,但又须要其余 bean 先实例化好,这个时候就须要使用 depends-on 特性了。depends-on 特性比较简单,就不演示了。仅贴一下配置文件的内容,以下:
这里有两个简单的类,其中 Hello 须要 World 在其以前完成实例化。相关配置以下:
1 2 |
<bean id="hello" class="xyz.coolblog.depnedson.Hello" depends-on="world"/> <bean id="world" class="xyz.coolblog.depnedson.World" /> |
BeanPostProcessor 是 bean 实例化时的后置处理器,包含两个方法,其源码以下:
1 2 3 4 5 6 7 8 |
public interface BeanPostProcessor { // bean 初始化前的回调方法 Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException; // bean 初始化后的回调方法 Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException; } |
BeanPostProcessor 是 Spring 框架的一个扩展点,经过实现 BeanPostProcessor 接口,咱们就可插手 bean 实例化的过程。好比你们熟悉的 AOP 就是在 bean 实例后期间将切面逻辑织入 bean 实例中的,AOP 也正是经过 BeanPostProcessor 和 IOC 容器创建起了联系。这里我来演示一下 BeanPostProcessor 的使用方式,以下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
/** * 日志后置处理器,将会在 bean 建立前、后打印日志 */ public class LoggerBeanPostProcessor implements BeanPostProcessor { @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { System.out.println("Before " + beanName + " Initialization"); return bean; } @Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { System.out.println("After " + beanName + " Initialization"); return bean; } } |
配置以下:
1 2 3 4 |
<bean class="xyz.coolblog.beanpostprocessor.LoggerBeanPostProcessor"/> <bean id="hello" class="xyz.coolblog.service.Hello"/> <bean id="world" class="xyz.coolblog.service.World"/> |
测试代码以下:
1 2 3 4 5 6 7 8 |
public class ApplicationContextTest { @Test public void testBeanPostProcessor() { String configLocation = "application-bean-post-processor.xml"; ApplicationContext applicationContext = new ClassPathXmlApplicationContext(configLocation); } } |
测试结果以下:
与 BeanPostProcessor 相似的还有一个叫 BeanFactoryPostProcessor 拓展点,顾名思义,用户能够经过这个拓展点插手容器启动的过程。不过这个不属于本系列文章范畴,暂时先不细说了。
Spring 中定义了一些列的 Aware 接口,好比这里的 BeanFactoryAware,以及 BeanNameAware 和 BeanClassLoaderAware 等等。经过实现这些 Aware 接口,咱们能够在运行时获取一些配置信息或者其余一些信息。好比实现 BeanNameAware 接口,咱们能够获取 bean 的配置名称(beanName)。经过实现 BeanFactoryAware 接口,咱们能够在运行时获取 BeanFactory 实例。关于 Aware 类型接口的使用,能够参考4.5.1 实现 ApplicationContextAware 接口
一节中的叙述,这里就不演示了。
我在去年八月份的时候,尝试过阅读 Spring 源码。不过 Spring 源码太多了,调用复杂,看着看着就迷失在了代码的海洋里。不过好在当时找到了一个通过简化后的类 Spring 框架,该框架黄亿华前辈在学习 Spring 源码时所写的 tiny-spring。若是你们以为看 Spring 源码比较困难,能够先学习一下 tiny-spring 的源码,先对 Spring 代码结构有个大概的了解。
另外也建议你们本身动手实现一个简单的 IOC 容器,经过实践,才会有更多的感悟。我在去年八月份的时候,实现过一个简单的 IOC 和 AOP(能够参考我去年发的文章:仿照 Spring 实现简单的 IOC 和 AOP - 上篇),并在最后将二者整合到了一块儿。正是有了以前的实践,才使得我对 Spring 的原理有了更进一步的认识。当作完一些准备工做后,再次阅读 Spring 源码,就没之前那么痛苦了。固然,Spring 的代码通过十几年的迭代,代码量很大。我在分析的过程当中也只是尽可能保证搞懂重要的逻辑,没法作到面面俱到。不过,若是你们愿意去读 Spring 源码,我相信会比我理解的更透彻。
除了上面说的动手实践外,在阅读源码的过程当中,若是实在看不懂,不妨调试一下。好比某个变量你不知道有什么用,可是它又比较关键,在多个地方都出现了,显然你必需要搞懂它。那么此时写点测试代码调试一下,立马就能知道它有什么用了。
以上是我在阅读源码时所使用过的一些方法,固然仅有上面那些可能还不够。本节的最后再推荐两本书,以下:
第二本书在豆瓣上的评分不太好,不过我以为还好。这本书我仅看了容器部分的代码分析,总的来讲还行。我在看循环依赖那一块的代码时,这本书仍是给了我一些帮助的。好了,本节就到这里。
在本文的最后一章,我来讲说我为何阅读 Spring 的源码吧。对我我的而言,有两个缘由。第一,做为 Java Web 开发人员,咱们基本上绕不过 Spring 技术栈。固然除了 Spring,还有不少其余的选择(好比有些公司本身封装框架)。但不能否认,Spring 如今还是主流。对于这样一个常常打交道的框架,弄懂实现原理,还有颇有必要的。起码在它出错输出一大堆异常时,你不会很心慌,能够从容的 debug。第二,我去年的这个时候,工做差很少快满一年。我在写第一年的工做总结时,以为很心慌。感受第一年好像只学到了一点开发的皮毛,技术方面,没什么积累。加上后来想换工做,内心想着就本身如今的水平恐怕要失业了。因此为了平复本身焦虑的情绪,因而在去年的八月份的时候,我开始写博客了。到如今写了近30篇博客(比较少),其中有两篇文章被博客平台做为优秀文章推荐过。如今,又快到7月份了,工龄立刻又要+1了。因此我如今在抓紧写 Spring 相关的文章,但愿在六月份多写几篇。算是对本身工做满两年的一个阶段性技术总结,也是一个记念吧。
固然,看懂源码并非什么很了不得的事情,毕竟写这些源码的人才是真正的大神。我在大学的时候,自学 MFC(没错,就是微软早已淘汰的东西)。并读过侯捷老师著的《深刻浅出MFC》一书,这本书中的一句话对我影响至深 - “勿在浮沙筑高台”。勿在浮沙筑高台,对于一个程序员来讲,基础很重要。因此我如今也在不断的学习,但愿能把基础打好,这样之后才能进入更高的层次。
好了,感谢你们耐心看完个人唠叨。本文先到这里,我要去写后续的文章了,后面再见。bye~
更新时间 | 标题 |
---|---|
2018-05-30 | Spring IOC 容器源码分析系列文章导读 |
2018-06-01 | Spring IOC 容器源码分析 - 获取单例 bean |
2018-06-04 | Spring IOC 容器源码分析 - 建立单例 bean 的过程 |
2018-06-06 | Spring IOC 容器源码分析 - 建立原始 bean 对象 |
2018-06-08 | Spring IOC 容器源码分析 - 循环依赖的解决办法 |
2018-06-11 | Spring IOC 容器源码分析 - 填充属性到 bean 原始对象 |
2018-06-11 | Spring IOC 容器源码分析 - 余下的初始化工做 |
更新时间 | 标题 |
---|---|
2018-06-17 | Spring AOP 源码分析系列文章导读 |
2018-06-20 | Spring AOP 源码分析 - 筛选合适的通知器 |
2018-06-20 | Spring AOP 源码分析 - 建立代理对象 |
2018-06-22 | Spring AOP 源码分析 - 拦截器链的执行过程 |
更新时间 | 标题 |
---|---|
2018-06-29 | Spring MVC 原理探秘 - 一个请求的旅行过程 |
2018-06-30 | Spring MVC 原理探秘 - 容器的建立过程 |