目录html
上篇已经对IoC容器的设计进行了分析(Spring源码阅读-IoC容器解析),本篇将对ApplicationContext
经典的继承层次图进行详细的分析,在心中造成一个大体的印象,以便后面一步步调试源码的时候,不会太眼花缭乱。让咱们一步步的前进吧...java
使用IDEA的继承层次工具生成以下的图(选中ApplicationContext --> Ctrl+H):web
(舒适提示:双击可查看高清大图)spring
从上图能很清楚的看出,ApplicationContext
的子接口分为两个部分:编程
ConfigurableApplicationContext
:大部分的应用上下文都实现了该接口WebApplicationContext
:在web应用程序中使用从上面的类的继承层次图能看到,ConfigurableApplicationContext
是比较上层的一个接口,该接口也是比较重要的一个接口,几乎全部的应用上下文都实现了该接口。该接口在ApplicationContext
的基础上提供了配置应用上下文的能力,此外提供了生命周期的控制能力。先看一下该接口的继承关系图(为了更加简洁,去掉了ApplicationContext
继承的接口):设计模式
(舒适提示:双击可查看高清大图)缓存
Closeable
接口用于关闭应用上下文,释放全部的资源和锁,这也包括摧毁全部缓存的单例的bean,常见的try-with-resources用法以下,执行完try体中的代码后会自动的调用close
方法:app
try (ConfigurableApplicationContext cac = ...) { // 编写代码 ... }
Lifecycle
定义了启动/中止生命周期的控制的一些方法,其中的方法以下:webapp
void start(); // 启动组件 void stop(); // 中止组件 boolean isRunning(); // 组件是否正在运行
接下来看一下ConfigurableApplicationContext
中的方法:ide
void setId(String id); // 设置应用上下文惟一的id void setParent(ApplicationContext parent); // 设置应用程序上下文的父级 void setEnvironment(ConfigurableEnvironment environment); // 设置应用上下文的环境 ConfigurableEnvironment getEnvironment(); // 获取应用上下文的环境 // 添加一个新的BeanFactoryPostProcessor void addBeanFactoryPostProcessor(BeanFactoryPostProcessor postProcessor); // 添加应用程序监听器 void addApplicationListener(ApplicationListener<?> listener); // 添加协议解析器,可能会覆盖默认的规则 void addProtocolResolver(ProtocolResolver resolver); // 加载或者刷新配置 void refresh() throws BeansException, IllegalStateException; // 向JVM runtime注册一个关闭钩子,JVM关闭时关闭这个上下文 void registerShutdownHook(); // 应用程序上问下是不是激活状态 boolean isActive(); // 获取应用上下文内部的bean factory ConfigurableListableBeanFactory getBeanFactory() throws IllegalStateException;
上面的这些方法基本上是提供了对某些特性的实现进行支撑的方法。
看了这么多方法,下面看一下ApplicationContext
的抽象的实现。
AbstractApplicationContext
是ApplicationContext
接口的抽象实现,这个抽象类仅仅是实现了公共的上下文特性。这个抽象类使用了模板方法设计模式,须要具体的实现类去实现这些抽象的方法。对相关接口的实现以下:
ApplicationContext
接口的实现ConfigurableApplicationContext
接口的实现BeanFactory
接口的实现ListableBeanFactory
接口的实现HierarchicalBeanFactory
接口的实现MessageSource
接口的实现ResourcePatternResolver
的实现Lifecycle
接口的实现本文不会详细的讲解这个类中的具体的实现细节,后面会有更加的详细的介绍。下面看下里面的抽象方法:
// 刷新BeanFactory,用于执行实际的配置加载,该方法在其余的初始化工做以前被refresh()方法调用 protected abstract void refreshBeanFactory() throws BeansException, IllegalStateException; // 关闭BeanFactory,用于释放内部使用的BeanFactory· protected abstract void closeBeanFactory(); // 获取内部使用的BeanFactory public abstract ConfigurableListableBeanFactory getBeanFactory() throws IllegalStateException;
那么对须要实现的方法通过抽象后,只剩下少许的须要子类去实现的方法。
GenericApplicationContext
继承自AbstractApplicationContext
,是为通用目的设计的,它能加载各类配置文件,例如xml,properties等等。它的内部持有一个DefaultListableBeanFactory
的实例,实现了BeanDefinitionRegistry
接口,以便容许向其应用任何bean的定义的读取器。为了可以注册bean的定义,refresh()
只容许调用一次。常见的使用以下:
GenericApplicationContext ctx = new GenericApplicationContext(); XmlBeanDefinitionReader xmlReader = new XmlBeanDefinitionReader(ctx); xmlReader.loadBeanDefinitions(new ClassPathResource("applicationContext.xml")); PropertiesBeanDefinitionReader propReader = new PropertiesBeanDefinitionReader(ctx); propReader.loadBeanDefinitions(new ClassPathResource("otherBeans.properties")); ctx.refresh(); MyBean myBean = (MyBean) ctx.getBean("myBean"); ..
这个类的实现没有太多须要注意的地方,须要注意的有两点:
DefaultListableBeanFactory
的实例,提供了一些方法来配置该实例,例如是否容许bean定义的覆盖、是否容许bean之间的循环应用等等。BeanDefinitionRegistry
,bean的定义注册。以便能经过BeanDefinitionReader
读取bean的配置,并注册。BeanDefinitionRegistry
接口的实现是直接使用内部的DefaultListableBeanFactory
的实例。GenericXmlApplicationContext
继承自GenericApplicationContext
,内置了对XML的支持。它很是的方便和灵活,是ClassPathXmlApplicationContext
和FileSystemXmlApplicationContext
的一种替代品。能够发现,它的内部有一个XmlBeanDefinitionReader
的实例,专门用于处理XML的配置。
StaticApplicationContext
继承自GenericApplicationContext
,主要用于编程式的注入bean和消息,而不是从外部的配置源读取bean的定义。主要是在测试时很是有用。经过阅读源代码能够看到,它的内部有一个StaticMessageSource
的实例,使用addMessage
方法添加消息。每次在编程式的注入bean时,都会建立一个GenericBeanDefinition
的实例。
ResourceAdapterApplicationContext
继承自GenericApplicationContext
,是为JCA(J2EE Connector Architecture)的ResourceAdapter设计的,主要用于传递BootstrapContext
的实例给实现了BootstrapContextAware
接口且由spring管理的bean。覆盖了postProcessBeanFactory
方法来实现此功能。
GenericGroovyApplicationContext
继承自GenericApplicationContext
,实现了GroovyObject
接口以便可以使用点的语法(.xx)取代getBean
方法来获取bean。它主要用于Groovy bean的定义,与GenericXmlApplicationContext
同样,它也能解析XML格式定义的bean。内部使用GroovyBeanDefinitionReader
来完成groovy脚本和XML的解析。
AnnotationConfigApplicationContext
继承自GenericApplicationContext
,提供了注解配置(例如:Configuration、Component、inject等)和类路径扫描(scan方法)的支持,可使用register(Class<?>... annotatedClasses)
来注册一个一个的进行注册。实现了AnnotationConfigRegistry接口,来完成对注册配置的支持,只有两个方法:register和scan。内部使用AnnotatedBeanDefinitionReader
来完成注解配置的解析,使用ClassPathBeanDefinitionScanner
来完成类路径下的bean定义的扫描。
AbstractRefreshableApplicationContext
继承自AbstractApplicationContext
,支持屡次进行刷新(屡次调用refresh
方法),每次刷新时在内部建立一个新的bean工厂的实例。子类仅仅须要实现loadBeanDefinitions
方法,该方法在每次刷新时都会调用。
AbstractRefreshableConfigApplicationContext
继承自AbstractRefreshableApplicationContext
,添加了对指定的配置文件路径的公共的处理,能够把他看做基于XML的应用上下文的基类。实现了以下的两个接口:
BeanNameAware
用于设置上下文的bean的名称,只有一个方法:void setBeanName(String name)
InitializingBean
用于上下文一切就绪后,若是还未刷新,那么就执行刷新操做,只有一个方法:void afterPropertiesSet()
AbstractXmlApplicationContext
继承自AbstractRefreshableConfigApplicationContext
,用于描绘包含能被XmlBeanDefinitionReader
所理解的bean定义的XML文档。子类只须要实现getConfigResources
和getConfigLocations
来提供配置文件资源。
FileSystemXmlApplicationContext
继承自AbstractXmlApplicationContext
,用于解析文件系统中XML配置文件。文件的路径能够是具体的文件路径,例如:xxx/application.xml,也能够是ant风格的配置,例如:xxx/*-context.xml。
看下面的经过文件路径来获取资源的代码:
@Override protected Resource getResourceByPath(String path) { if (path.startsWith("/")) { path = path.substring(1); } return new FileSystemResource(path); }
文件路径前面的/
会被去掉,不管是否路径前面是否加上/
,文件路径都会解析成相对路径,即基于JVM的当前工做路径。获取到的资源对象是FileSystemResource
,用于处理文件系统相关的资源。
ClassPathXmlApplicationContext
继承自AbstractXmlApplicationContext
,和FileSystemXmlApplicationContext
相似,只不过ClassPathXmlApplicationContext
是用于处理类路径下的XML配置文件。文件的路径能够是具体的文件路径,例如:xxx/application.xml,也能够是ant风格的配置,例如:xxx/*-context.xml。
该接口提供了在web应用中的配置,接口提供了一个ServletContext getServletContext()
用来获取ServletContext
对象。该接口会和ServletContext的一个属性进行绑定,这个属性就是ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE
。定义了三个做用域的名称:SCOPE_REQUEST
,SCOPE_SESSION
,SCOPE_APPLICATION
。在工厂中的bean的名称:SERVLET_CONTEXT_BEAN_NAME
。ServletContext初始化参数名称:CONTEXT_PARAMETERS_BEAN_NAME
。在工厂中ServletContext属性值环境bean的名称:CONTEXT_ATTRIBUTES_BEAN_NAME
。
ConfigurableWebApplicationContext
继承自WebApplicationContext
和ConfigurableApplicationContext
,提供了web应用上下文的可配置的能力。相关接口定义以下:
// 设置web应用上下文的ServletContext void setServletContext(@Nullable ServletContext servletContext); // 设置web应用上下文的ServletConfig void setServletConfig(@Nullable ServletConfig servletConfig); // 获取web应用上下文的ServletConfig ServletConfig getServletConfig(); // 设置web应用上下文的命名空间 void setNamespace(@Nullable String namespace); // 获取web应用上下文的命名空间 String getNamespace(); // 以初始化参数的形式设置web应用上下文的配置文件位置 void setConfigLocation(String configLocation); // 设置web应用上下文的配置文件的位置 void setConfigLocations(String... configLocations); // 获取web应用上下文的配置文件位置 String[] getConfigLocations();
上面的接口主要都是一些设置或者获取的方法,在web应用上下文中须要用到的一些东西。
GenericWebApplicationContext
继承自GenericApplicationContext
,实现了ConfigurableWebApplicationContext
和ThemeSource
接口。该类设计的目的不是在web.xml中进行声明式的安装,而是编程式的安装,例如使用WebApplicationInitializers
来构建内嵌的上下文。该接口在ConfigurableWebApplicationContext
的内容都是一个伪实现,调用其中的大多数方法都会抛出异常。你也许注意到了,他实现了ThemeSource
接口,那么他有什么用呢?字面意思是主题源,它设计的目的主要是用于消息的国际化。
StaticWebApplicationContext
继承自StaticApplicationContext
实现了ConfigurableWebApplicationContext
和ThemeSource
接口。该接口主要是用在测试的环境,不用于产品环境。
GenericWebApplicationContext
继承自AbstractRefreshableConfigApplicationContext
,实现了ConfigurableWebApplicationContext
和ThemeSource
接口,主要是用于web环境下。在web程序启动的时候,提供一个configLocations
属性,经过ConfigurableWebApplicationContext
接口来进行填充。子类化这个接口是很简单的,全部你所须要作的事情就是实现loadBeanDefinitions
方法,来实现你本身的bean定义的加载逻辑。
XmlWebApplicationContext
继承自AbstractRefreshableWebApplicationContext
,接受能被XmlBeanDefinitionReader
所理解的XML文档配置。对于根上下文,默认的配置文件路径是/WEB-INF/applicationContext.xml
,对于命名空间为test-servlet的上下文,默认的配置文件路径是/WEB-INF/test-servlet.xml
(就像servlet-name为test的DispatcherServlet实例)。
默认的配置文件路径处理的代码以下:
protected String[] getDefaultConfigLocations() { if (getNamespace() != null) { return new String[] {DEFAULT_CONFIG_LOCATION_PREFIX + getNamespace() + DEFAULT_CONFIG_LOCATION_SUFFIX}; } else { return new String[] {DEFAULT_CONFIG_LOCATION}; } }
和其余的上下文同样,bean定义的加载也是在void loadBeanDefinitions(DefaultListableBeanFactory beanFactory)
方法中,使用的是XmlBeanDefinitionReader
。
GroovyWebApplicationContext
继承自AbstractRefreshableWebApplicationContext
,实现了GroovyObject
接口,接受能被GroovyBeanDefinitionReader
所理解的groovy bean定义脚本和XML文档配置。对于web环境,基本上是和GenericGroovyApplicationContext
是等价的。对于根上下文,默认的配置文件路径是/WEB-INF/applicationContext.groovy
,对于命名空间为test-servlet的上下文,默认的配置文件路径是/WEB-INF/test-servlet.xml
(就像servlet-name为test的DispatcherServlet实例)。
默认的配置文件路径处理的代码以下:
protected String[] getDefaultConfigLocations() { if (getNamespace() != null) { return new String[] {DEFAULT_CONFIG_LOCATION_PREFIX + getNamespace() + DEFAULT_CONFIG_LOCATION_SUFFIX}; } else { return new String[] {DEFAULT_CONFIG_LOCATION}; } }
和其余的上下文同样,bean定义的加载也是在void loadBeanDefinitions(DefaultListableBeanFactory beanFactory)
方法中,使用的是GroovyBeanDefinitionReader
。
AnnotationConfigWebApplicationContext
继承自AbstractRefreshableWebApplicationContext
,
接受注解的类做为输入(特殊的@Configuration注解类,通常的@Component注解类,与JSR-330兼容的javax.inject注解)。容许一个一个的注入,一样也能使用类路径扫描。对于web环境,基本上是和AnnotationConfigApplicationContext
等价的。使用AnnotatedBeanDefinitionReader
来对注解的bean进行处理,使用ClassPathBeanDefinitionScanner
来对类路径下的bean进行扫描。
部分代码以下:
protected void loadBeanDefinitions(DefaultListableBeanFactory beanFactory) { AnnotatedBeanDefinitionReader reader = getAnnotatedBeanDefinitionReader(beanFactory); ClassPathBeanDefinitionScanner scanner = getClassPathBeanDefinitionScanner(beanFactory); ... if (!this.annotatedClasses.isEmpty()) { .... reader.register(ClassUtils.toClassArray(this.annotatedClasses)); } if (!this.basePackages.isEmpty()) { .... scanner.scan(StringUtils.toStringArray(this.basePackages)); } String[] configLocations = getConfigLocations(); if (configLocations != null) { for (String configLocation : configLocations) { try { Class<?> clazz = ClassUtils.forName(configLocation, getClassLoader()); if (logger.isTraceEnabled()) { logger.trace("Registering [" + configLocation + "]"); } reader.register(clazz); } catch (ClassNotFoundException ex) { .... } } } } }