1 前言

众所周知,Spring可以帮我们管理我们需要的bean。在我们需要用到这些bean的时候,可以很方便的获取到它,然后进行一系列的操作。比如,我们定义一个bean MyTestBean

public class MyTestBean {
private String testStr = "testStr";
public String getTestStr() {
return testStr;
}
public void setTestStr(String testStr) {
this.testStr = testStr;
}

然后xml配置一下

<?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-3.1.xsd">
<bean id = "myTestBean" class="bean.MyTestBean"/>
</beans>

编写一下测试代码,测试一下,就会看到测试通过的结果。

public class BeanFactoryTest {
@Test
public void testSimpleLoad() {
BeanFactory bf = new XmlBeanFactory(new ClassPathResource("beanFactoryTest.xml"));
MyTestBean bean = (MyTestBean)bf.getBean("myTestBean");
assert "testStr".equals(bean.getTestStr());
}
}

直接使用BeanFactory作为容器对于Spring来说不常见,这里只是用来测试,以便可以更快更好地分析Spring内部原理。其涉及到的一些组件,贯穿整个Spring容器当中,对于我们了解其他Spring容器也有很大的帮助。限于篇幅,这里只介绍该容器创建的部分,即对于new XmlBeanFactory(new ClassPathResource(“beanFactoryTest.xml”)),Spring都干了些什么。

2 Spring容器创建原理

2.1 整体实现流程

首先我们大致了解一下Spring容器创建的整体过程。

整体时序图

可以看到,该Spring容器创建大致分为以下几部分:

  1. 资源的封装,以Resource封装配置文件
  2. 加载BeanDefinition
  3. 解析配置文件,获取Document
  4. 解析及注册BeanDefinition
  5. 标签的解析,分为默认标签和自定义标签的解析

下面我们就以这样的顺序对各个部分从代码实现上进行具体分析。

2.2 核心类介绍

在进行具体创建逻辑之前,我们先对Spring容器创建的核心类进行介绍,以便我们更好地掌握它的实现过程。

2.2.1 DefaultListableBeanFactory

XmlBeanFactory继承自DefaultListableBeanFactory,而DefaultListableBeanFactory是整个bean加载的核心部分,是Spring注册及加载bean的默认实现。XmlBeanFactory与DefaultListableBeanFactory的不同之处在于,XmlBeanFactory使用了自定义的XML读取器XmlBeanDefinitionReader,实现了个性化的BeanDefinitionReader读取。

容器加载相关类图

2.2.2 XmlBeanDefinitionReader

XmlBeanDefinitionReader用于资源文件读取、解析及Bean注册。读取Xml配置文件的流程大致为,首先使用ResourceLoader将资源文件路径转换为对应的Resource文件,然后将Resource文件转换为Document文件,最后对Document及Element进行解析。

配置文件读取相关类图

2.2.3 BeanDefinition

在Spring中,BeanDefinition是配置文件元素标签在容器中的内部表示形式,包含了元素的所有信息。Spring将配置文件中的转换为BeanDefinition,并将这些BeanDefinition注册到BeanDefinitionRegistry中。BeanDefinitionRegistry以map形式保存,后续操作直接从BeanDefinitionRegistry中读取配置信息。

BeanDefinition及其实现类

2.3 配置文件的封装

在Java中不同的资源都要抽象成URL,然后使用不同类型的URLStreamHandler处理不同的URL表示的资源。但是,Spring对其内部使用到的资源实现了自己的抽象结构:Resource接口封装底层资源。主要原因有3点:

  1. URL没有默认定义相对Classpath或ServletContext等资源的handler
  2. URL没有提供基本的方法,例如检查当前资源是否存在是否可读等
  3. 自定义URL handler需要了解URL实现机制

对不同来源的资源文件都有相应的Resource实现:文件(FileSystemResource)、Classpath资源(ClassPathResource)、URL资源(UrlResource)、InputStream资源(InputStreamResource)、Byte数组(ByteArrayResource)等。

资源文件处理相关类图

2.4 加载BeanDefinition

下面我们就从代码层次看看整个容器究竟是怎么实现的。观察测试代码,我们可以将XmlBeanFactory的构造方法作为切入点进行分析。

public XmlBeanFactory(Resource resource) throws BeansException {
this(resource, null);
}
public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException {
super(parentBeanFactory);
// 加载BeanDefinition
this.reader.loadBeanDefinitions(resource);
}

主要做了两件事,一是调用父类的构造方法,二是加载BeanDefinition。
首先我们先进入父类构造方法,最终进到AbstractAutowireCapableBeanFactory构造方法中。

public AbstractAutowireCapableBeanFactory() {
super();
// 忽略BeanNameAware、BeanFactoryAware和BeanClassLoaderAware接口的自动装配功能
ignoreDependencyInterface(BeanNameAware.class);
ignoreDependencyInterface(BeanFactoryAware.class);
ignoreDependencyInterface(BeanClassLoaderAware.class);
}

主要是ignoreDependencyInterface方法,它的主要功能是忽略给定接口的自动装配功能。实现上很简单,就是把这些Class加入到ignoredDependencyInterfaces集合中,ignoredDependencyInterfaces是Set<class<?>>类型。

再看加载Bean的方法,执行的是XmlBeanDefinitionReader类的loadBeanDefinitions方法。进入方法,可以看到主要就是做了两件事,一是构造InputSource,这个类全路径名是org.xml.sax.InputSource,这步的目的就是通过SAX读取XML文件事先准备一下InputSource对象。而真正的加载Bean的逻辑在doLoadBeanDefinitions方法中。

public int loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException {
InputStream inputStream = encodedResource.getResource().getInputStream();
// 构建InputSource,用于解析XML
InputSource inputSource = new InputSource(inputStream);
if (encodedResource.getEncoding() != null) {
inputSource.setEncoding(encodedResource.getEncoding());
}
// 实际加载BeanDefinition的执行逻辑
return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
}

doLoadBeanDefinitions方法首先加载XML文件,得到Document对象,然后根据Document对象注册Bean。我们首先看下得到Document对象的过程。

2.5 获取Document

获取Document,首先通过getValidationModeForResource获取XML验证模式,然后解析得到Document对象。

protected Document doLoadDocument(InputSource inputSource, Resource resource) throws Exception {
// 首先获取XML验证模式,然后SAX方式解析得到Document
return this.documentLoader.loadDocument(inputSource, getEntityResolver(), this.errorHandler,
getValidationModeForResource(resource), isNamespaceAware());
}

常用的XML验证模式有两种:DTD和XSD,实际使用哪种验证模式在getValidationModeForResource中进行了解析。这个方法判断是DTD验证还是XSD验证,仅仅是判断一下XML是否包含DOCTYPE字符串。

而解析得到Document的方法很简单,就是通过SAX解析XML文档的套路。这里不再赘述。

2.6 解析及注册BeanDefinition

当把文件转换为Document后,接下来的提取及注册bean就是我们的重头戏了。调用了以下方法。

public int registerBeanDefinitions(Document doc, Resource resource) throws BeanDefinitionStoreException {
BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();
int countBefore = getRegistry().getBeanDefinitionCount();
// 注册BeanDefinition
documentReader.registerBeanDefinitions(doc, createReaderContext(resource));
return getRegistry().getBeanDefinitionCount() - countBefore;
}

在这个方法中很好地应用了单一职责原则,将逻辑处理委托给单一的类进行处理,而这个逻辑处理类就是BeanDefinitionDocumentReader。BeanDefinitionDocumentReader是一个接口,实例化的工作在createBeanDefinitionDocumentReader()中完成,真正的类型是DefaultBeanDefinitionDocumentReader。而它的registerBeanDefinitions方法很简单,仅仅是先根据Document获取了root,实际注册在doRegisterBeanDefinitions方法中。

protected void doRegisterBeanDefinitions(Element root) {
if (this.delegate.isDefaultNamespace(root)) {
String profileSpec = root.getAttribute(PROFILE_ATTRIBUTE);
if (StringUtils.hasText(profileSpec)) {
String[] specifiedProfiles = StringUtils.tokenizeToStringArray(
profileSpec, BeanDefinitionParserDelegate.MULTI_VALUE_ATTRIBUTE_DELIMITERS);
// 如果环境变量不包含指定profile,则流程结束
if (!getReaderContext().getEnvironment().acceptsProfiles(specifiedProfiles)) {
return;
}
}
}
preProcessXml(root);
// 解析BeanDefinition
parseBeanDefinitions(root, this.delegate);
postProcessXml(root);
}

注册过程首先对profile进行处理,如果是环境变量定义的则进行处理,否则不进行处理。然后就是解析bean。这里调用了preProcessXml(root)和postProcessXml(root)两个方法,但是发现这两个方法是空方法。这里应用了模板方法模式。如果继承自DefaultBeanDefinitionDocumentReader的子类需要在Bean解析前后做一些处理的话,只需要重写这两个方法即可。

protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {
if (delegate.isDefaultNamespace(root)) {
// 根元素是默认命名空间
NodeList nl = root.getChildNodes();
for (int i = 0; i < nl.getLength(); i++) {
Node node = nl.item(i);
if (node instanceof Element) {
Element ele = (Element) node;
if (delegate.isDefaultNamespace(ele)) {
// 如果元素是默认命名空间,则按默认方式解析元素
parseDefaultElement(ele, delegate);
}
else {
// 如果这个元素是自定义命名空间,则按自定义方式解析元素
delegate.parseCustomElement(ele);
}
}
}
}
else {
// 如果根元素是自定义命名空间,则按自定义方式解析元素
delegate.parseCustomElement(root);
}
}

而解析bean需要判断元素是否是默认命名空间,如果是则调用parseDefaultElement(ele, delegate)方法,不是则调用delegate.parseCustomElement(ele)方法。判断是否是默认命名空间,调用isDefaultNamespace方法,元素或者节点的命名空间与Spring中固定的命名空间http://www.springframework.org/schema/beans 进行对比,一致则认为是默认的,否则就认为是自定义的。

2.7 默认标签的解析

默认标签的解析逻辑一目了然,分别对4种不同标签(import、alias、bean和beans)做了不同的处理。

2.7.1 bean标签的解析及注册

对bean标签的解析是通过processBeanDefinition(ele, delegate)方法进行的。大致逻辑总结如下:

  1. 首先委托BeanDefinitionDelegate类的parseBeanDefinitionElement方法进行元素解析,返回BeanDefinitionHolder类型的实例bdHolder,经过这个方法后,bdHolder实例已经包含我们配置文件中配置的各种属性了,例如class、name、id、alias等属性。
  2. 如果bdHolder不为空,若存在默认标签的子节点下再有自定义属性,还需要再次对自定义标签进行解析。
  3. 解析后,对bdHolder进行注册。注册操作委托给了BeanDefinitionReaderUtils的registerBeanDefinition方法。
  4. 发出响应事件,通知相关监听器,这个bean已经加载完了。
protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {
// 解析元素信息,用bdHolder封装
BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
if (bdHolder != null) {
// 如果标签下有自定义属性,则对自定义属性进行解析
bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);
// 对bdHolder进行注册
BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());
// 发送注册事件给监听器.
getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));
}
}

1)元素解析及信息提取

首先我们从元素解析及信息提取开始:delegate.parseBeanDefinitionElement(ele)。进入BeanDefinitionParserDelegate类parseBeanDefinitionElement方法,主要完成如下内容:

  1. 提取元素的id和name属性
  2. 解析其他所有属性,封装到GenericBeanDefinition类型的实例中,对应this.parseBeanDefinitionElement(ele, beanName, containingBean)方法
  3. 如果bean没有指定beanName,则使用默认规则为此Bean生成beanName
  4. 将获取到的信息封装到BeanDefinitionHolder实例中

beanName取值策略是,首先取id,如果没有指定id则取name[0](因为name可以指定多个),如果name也没有指定,则采取自动生成方式生成。

最终bean元素的所有属性和子元素信息都保存到GenericBeanDefinition中了。至此就完成了XML文档到GenericBeanDefinition的转换。

2)默认标签中自定义标签元素的解析

如果这个bean使用的是默认的标签配置,但是其中的子元素却使用了自定义配置,这时这部分内容就起作用了。入口是delegate.decorateBeanDefinitionIfRequired(ele, bdHolder)方法。它分别对元素的所有属性和子元素进行了decorateIfRequired方法的调用。decorateIfRequired方法会判断,如果是自定义节点,则找出自定义类型所对应的NamespaceHandler并进行进一步解析。

public BeanDefinitionHolder decorateIfRequired(
Node node, BeanDefinitionHolder originalDef, BeanDefinition containingBd) {
String namespaceUri = getNamespaceURI(node);
if (!isDefaultNamespace(namespaceUri)) {
// 如果元素是自定义元素,则根据命名空间找到对应的命名空间处理器
NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
if (handler != null) {
// 自定义命名空间处理器处理bdHolder
return handler.decorate(node, originalDef, new ParserContext(this.readerContext, this, containingBd));
}
}
return originalDef;
}

3)BeanDefinition的注册

BeanDefinition注册分为通过beanName注册BeanDefinition和注册别名两部分。进入BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, this.getReaderContext().getRegistry())方法内部,如下

public static void registerBeanDefinition(
BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)
throws BeanDefinitionStoreException {
String beanName = definitionHolder.getBeanName();
// 通过beanName注册BeanDefinition
registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition());
String[] aliases = definitionHolder.getAliases();
if (aliases != null) {
for (String alias : aliases) {
// 注册别名
registry.registerAlias(beanName, alias);
}
}
}

通过beanName注册BeanDefinition主要进行了4个步骤:

  1. 对AbstractBeanDefinition的校验,主要是对于AbstractBeanDefinition的methodOverrides属性的
  2. 对于beanName已经注册的情况的处理,如果已经设置了不允许bean的覆盖,会抛出异常,否则进行覆盖
  3. 加入Map缓存,beanName为key,BeanDefinition为value
  4. 清除之前留下的对应beanName的缓存
public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition) {
if (beanDefinition instanceof AbstractBeanDefinition) {
// 对AbstractBeanDefinition的校验
((AbstractBeanDefinition) beanDefinition).validate();
}
BeanDefinition oldBeanDefinition = this.beanDefinitionMap.get(beanName);
if (oldBeanDefinition != null) {
if (!isAllowBeanDefinitionOverriding()) {
// 如果beanName已经注册,并且设置了不允许bean覆盖,会抛出异常
throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(), beanName,
"Cannot register bean definition [" + beanDefinition + "] for bean '" + beanName +
"': There is already [" + oldBeanDefinition + "] bound.");
}
// 将beanDefinition存到Map,beanName为key
this.beanDefinitionMap.put(beanName, beanDefinition);
}
else {
if (hasBeanCreationStarted()) {
synchronized (this.beanDefinitionMap) {
// 将beanDefinition存到Map,beanName为key
this.beanDefinitionMap.put(beanName, beanDefinition);
}
}
else {
// 将beanDefinition存到Map,beanName为key
this.beanDefinitionMap.put(beanName, beanDefinition);
}
}
if (oldBeanDefinition != null || containsSingleton(beanName)) {
// 清除之前留下的对应beanName的缓存
resetBeanDefinition(beanName);
}
}

注册别名的原理相对简单,分为4个步骤:

  1. alias和beanName相同情况处理,此时会删除原有的alias
  2. alias覆盖处理,若aliasName已经使用了并已经指向了另一个beanName,且设置了别名不能覆盖,则会抛出异常
  3. alias循环检查,如果出现了别名循环的情况,则抛出异常
  4. 注册alias
public void registerAlias(String name, String alias) {
if (alias.equals(name)) {
this.aliasMap.remove(alias);
} else {
String registeredName = (String)this.aliasMap.get(alias);
if (registeredName != null) {
if (registeredName.equals(name)) {
return;
}
// 若aliasName已经使用了并已经指向了另一个beanName,且设置了别名不能覆盖,则会抛出异常
if (!this.allowAliasOverriding()) {
throw new IllegalStateException("Cannot register alias '" + alias + "' for name '" + name + "': It is already registered for name '" + registeredName + "'.");
}
}
// alias循环检查
this.checkForAliasCircle(name, alias);
// alias注册
this.aliasMap.put(alias, name);
}
}

4)通知监听器解析及注册完成

通过代码this.getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder))完成此功能。这里的实现只是为了扩展,当需要对注册BeanDefinition事件进行监听时可以通过注册监听器的方式并将处理逻辑写入监听器中,目前Spring没有对此事件进行任何处理。

2.7.2 alias标签的解析

对alias标签的解析是通过processAliasRegistration(ele)方法处理的。

protected void processAliasRegistration(Element ele) {
String name = ele.getAttribute(NAME_ATTRIBUTE);
String alias = ele.getAttribute(ALIAS_ATTRIBUTE);
boolean valid = true;
if (!StringUtils.hasText(name)) {
valid = false;
}
if (!StringUtils.hasText(alias)) {
valid = false;
}
if (valid) {
// alias标签解析
getReaderContext().getRegistry().registerAlias(name, alias);
// 通知监听器
getReaderContext().fireAliasRegistered(name, alias, extractSource(ele));
}
}

进入方法内部可以看到,this.getReaderContext().getRegistry().registerAlias(name, alias)方法实现了alias标签的解析,而这个方法实际就是前面注册别名的那个方法。this.getReaderContext().fireAliasRegistered(name, alias, this.extractSource(ele))这个方法用于在别名注册后通知监听器做相应的处理,这里的实现只是为了扩展,目前Spring没有对此事件进行任何处理。

2.7.3 import标签的解析

对import标签的解析,Spring大致分为以下步骤:

  1. 获取import标签的resource属性配置的路径
  2. 解析路径中的系统属性,格式如“${user.dir}”,对应方法this.getReaderContext().getEnvironment().resolveRequiredPlaceholders(location)
  3. 判断resource属性配置的路径是绝对路径还是相对路径
  4. 如果是绝对路径,则调用bean的解析过程进行解析
  5. 如果是相对路径则计算出绝对路径后进行解析
  6. 通知监听器,解析完成

不管是绝对路径下import标签的解析还是相对路径下import标签的解析,通过跟踪代码发现,最后都会调到XmlBeanDefinitionReader类的loadBeanDefinitions方法,而这个方法在加载bean部分已经了解了。

2.7.4 嵌入式beans标签的解析

该标签解析调用的是DefaultBeanDefinitionDocumentReader的doRegisterBeanDefinitions方法,而这个方法已经在解析及注册BeanDefinitions部分了解了。嵌入式beans标签和非嵌入式beans标签的解析过程其实是一样的。

2.8 自定义标签的解析

自定义标签非常有用,我们熟知的标签就是采用自定义标签的原理实现的。下面来探究一下自定义标签的解析原理。
前面提到过解析自定义标签的入口,查看以下具体实现:

public BeanDefinition parseCustomElement(Element ele, BeanDefinition containingBd) {
String namespaceUri = this.getNamespaceURI(ele);
NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
if (handler == null) {
return null;
} else {
return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));
}
}

这里传入的containingBd为null。可以看到自定义标签解析的思路特别简单。无非是根据标签元素获取对应的命名空间,根据命名空间解析对应的处理器,然后根据用户自定义的处理器进行解析。

2.8.1 解析自定义标签处理器

通过元素可以获取它的命名空间,有了命名空间就可以进行NamespaceHandler提取了。在readerContext初始化的时候其属性namespaceHandlerResolver被初始化为DefaultNamespaceHandlerResolver的实例。所以调用resolve方法实际调用的是DefaultNamespaceHandlerResolver的方法。

public NamespaceHandler resolve(String namespaceUri) {
// 获取命名空间到处理器的映射关系
Map<String, Object> handlerMappings = getHandlerMappings();
Object handlerOrClassName = handlerMappings.get(namespaceUri);
if (handlerOrClassName == null) {
return null;
}
else if (handlerOrClassName instanceof NamespaceHandler) {
return (NamespaceHandler) handlerOrClassName;
}
else {
String className = (String) handlerOrClassName;
Class<?> handlerClass = ClassUtils.forName(className, this.classLoader);
if (!NamespaceHandler.class.isAssignableFrom(handlerClass)) {
throw new FatalBeanException("Class [" + className + "] for namespace [" + namespaceUri +
"] does not implement the [" + NamespaceHandler.class.getName() + "] interface");
}
// 实例化命名空间处理器
NamespaceHandler namespaceHandler = (NamespaceHandler) BeanUtils.instantiateClass(handlerClass);
namespaceHandler.init();
handlerMappings.put(namespaceUri, namespaceHandler);
return namespaceHandler;
}
}
private Map<String, Object> getHandlerMappings() {
if (this.handlerMappings == null) {
synchronized (this) {
if (this.handlerMappings == null) {
// 加载配置文件META-INF/spring.handlers
Properties mappings =
PropertiesLoaderUtils.loadAllProperties(this.handlerMappingsLocation, this.classLoader);
Map<String, Object> handlerMappings = new ConcurrentHashMap<String, Object>(mappings.size());
CollectionUtils.mergePropertiesIntoMap(mappings, handlerMappings);
this.handlerMappings = handlerMappings;
}
}
}
return this.handlerMappings;
}

可以看到自定义标签处理器的解析流程,首先通过this.getHandlerMappings()方法解析配置文件获取命名空间到处理器的映射关系,Map保存。然后实例化该命名空间处理器,调用init()初始化方法。而定义的this.handlerMappingsLocation变量在调用构造方法的时候代码写死了,是META-INF/spring.handlers。这个配置文件是用户自己去编写的,定义命名空间到处理器类的映射。命名空间处理器类的实现也是需要用户去实现,用户可以继承NamespaceHandlerSupport抽象类实现一下init()抽象方法。

2.8.2 标签解析

我们已经得到了由哪个标签处理器进行处理,接下来标签解析由handler.parse(ele, new ParserContext(this.readerContext, this, containingBd))去实现。

public BeanDefinition parse(Element element, ParserContext parserContext) {
// 获取元素解析器,进行解析
return findParserForElement(element, parserContext).parse(element, parserContext);
}
private BeanDefinitionParser findParserForElement(Element element, ParserContext parserContext) {
// 根据节点名称获取parser
String localName = parserContext.getDelegate().getLocalName(element);
BeanDefinitionParser parser = this.parsers.get(localName);
return parser;
}

在父类NamespaceHandlerSupport中可以看到,解析功能首先找到解析器,然后进行解析。查找解析器首先获取节点名称,然后通过Map parsers获取对应节点的解析器。而这个Map的赋值一般在用户实现的命名空间处理器init()方法中调用。

而自定义标签的解析任务由parse方法完成。可以看到,首先通过parseInternal方法将标签元素转换成了BeanDefinition,然后解析id和name属性并用BeanDefinitionHolder封装元素信息,接着进行BeanDefinition的注册,最后通知监听器。

public final BeanDefinition parse(Element element, ParserContext parserContext) {
// 解析元素
AbstractBeanDefinition definition = this.parseInternal(element, parserContext);
if (definition != null && !parserContext.isNested()) {
// 解析id
String id = this.resolveId(element, definition, parserContext);
String[] aliases = null;
if (this.shouldParseNameAsAliases()) {
// 解析name
String name = element.getAttribute("name");
if (StringUtils.hasLength(name)) {
aliases = StringUtils.trimArrayElements(StringUtils.commaDelimitedListToStringArray(name));
}
}
// holder封装元素信息
BeanDefinitionHolder holder = new BeanDefinitionHolder(definition, id, aliases);
// 注册
this.registerBeanDefinition(holder, parserContext.getRegistry());
if (this.shouldFireEvents()) {
BeanComponentDefinition componentDefinition = new BeanComponentDefinition(holder);
this.postProcessComponentDefinition(componentDefinition);
// 通知监听器
parserContext.registerComponent(componentDefinition);
}
}
return definition;
}

对于后面三步前面都介绍过了,只需看下parseInternal方法逻辑。

protected final AbstractBeanDefinition parseInternal(Element element, ParserContext parserContext) {
BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition();
// 解析parentName
String parentName = this.getParentName(element);
if (parentName != null) {
builder.getRawBeanDefinition().setParentName(parentName);
}
// 解析beanClass
Class<?> beanClass = this.getBeanClass(element);
if (beanClass != null) {
builder.getRawBeanDefinition().setBeanClass(beanClass);
} else {
String beanClassName = this.getBeanClassName(element);
if (beanClassName != null) {
builder.getRawBeanDefinition().setBeanClassName(beanClassName);
}
}
// 解析source
builder.getRawBeanDefinition().setSource(parserContext.extractSource(element));
if (parserContext.isNested()) {
builder.setScope(parserContext.getContainingBeanDefinition().getScope());
}
// 解析lazyInit
if (parserContext.isDefaultLazyInit()) {
builder.setLazyInit(true);
}
this.doParse(element, parserContext, builder);
return builder.getBeanDefinition();
}
protected void doParse(Element element, ParserContext parserContext, BeanDefinitionBuilder builder) {
this.doParse(element, builder);
}
protected void doParse(Element element, BeanDefinitionBuilder builder) {
}

可以看到,parseInternal方法实际就是先解析parentName、beanClass、source、scope和lazyInit,以BeanDefinition封装。然后调用doParse方法,这个方法也是用户自定义解析器需要实现的方法。最后返回这个BeanDefinition。

3 总结

至此,我们一步一步地分析了Spring容器创建基本原理的所有内容。没想到短短一句new XmlBeanFactory(new ClassPathResource(“beanFactoryTest.xml”)),Spring做了这么多事:配置文件的封装、Document获取和解析及注册BeanDefinition等。解析及注册BeanDefinition对默认标签和自定义标签进行不同的处理。对于默认标签,又分别对bean标签、alias标签、import标签和beans标签进行不同的处理。其中bean标签解析逻辑最为复杂也最为基础,而剩下的那几个标签又复用了bean标签处理的部分逻辑。

我们从中不仅可以学到Spring容器创建的基本原理,还可以学到许多编码规范及技巧,了解到好的代码是什么样子的。比如其中应用到的单一职责原则、模板方法模式等等。而且还可以发现,它的代码逻辑很清晰,往往通过它的方法名称就知道这个方法的功能,并且每个方法也不会特别长,增加了代码的可读性和可维护性。并且代码封装性很好,很多复杂的功能代码都可以复用。

在我们之后的开发工作中需要不断学习好的编码技巧及规范,应用到日常开发工作当中,最终形成我们自己的编码技巧及风格。

4 参考资料

《Spring技术内幕:深入解析Spring架构与设计原理》
《Spring源码深度解析》

作者:曹铭(大件技术工坊)

源码学习之Spring容器创建原理的更多相关文章

  1. Spring源码学习之IOC容器实现原理(一)-DefaultListableBeanFactory

    从这个继承体系结构图来看,我们可以发现DefaultListableBeanFactory是第一个非抽象类,非接口类.实际IOC容器.所以这篇博客以DefaultListableBeanFactory ...

  2. 框架源码系列八:Spring源码学习之Spring核心工作原理(很重要)

    目录:一.搞清楚ApplicationContext实例化Bean的过程二.搞清楚这个过程中涉及的核心类三.搞清楚IOC容器提供的扩展点有哪些,学会扩展四.学会IOC容器这里使用的设计模式五.搞清楚不 ...

  3. spring源码学习笔记之容器的基本实现(一)

    前言 最近学习了<<Spring源码深度解析>>受益匪浅,本博客是对学习内容的一个总结.分享,方便日后自己复习或与一同学习的小伙伴一起探讨之用. 建议与源码配合使用,效果更嘉, ...

  4. spring源码学习之路---IOC实现原理(三)

    作者:zuoxiaolong8810(左潇龙),转载请注明出处,特别说明:本博文来自博主原博客,为保证新博客中博文的完整性,特复制到此留存,如需转载请注明新博客地址即可. 上一章我们已经初步认识了Be ...

  5. 【spring源码学习】spring的IOC容器在初始化bean过程

    [一]初始化IOC的bean的时候Spring会执行的一些回调方法 (1)spring bean创建的前置处理 =>ApplicationContextAwareProcessor 在创建bea ...

  6. 框架源码系列六:Spring源码学习之Spring IOC源码学习

    Spring 源码学习过程: 一.搞明白IOC能做什么,是怎么做的  1. 搞明白IOC能做什么? IOC是用为用户创建.管理实例对象的.用户需要实例对象时只需要向IOC容器获取就行了,不用自己去创建 ...

  7. spring源码学习(三)--spring循环引用源码学习

    在spring中,是支持单实例bean的循环引用(循环依赖)的,循环依赖,简单而言,就是A类中注入了B类,B类中注入了A类,首先贴出我的代码示例 @Component public class Add ...

  8. 结合源码浅谈Spring容器与其子容器Spring MVC 冲突问题

    容器是整个Spring 框架的核心思想,用来管理Bean的整个生命周期. 一个项目中引入Spring和SpringMVC这两个框架,Spring是父容器,SpringMVC是其子容器,子容器可以看见父 ...

  9. Springcloud源码学习笔记1—— Zuul网关原理

    系列文章目录和关于我 源码基于 spring-cloud-netflix-zuul-2.2.6.RELEASE.jar 需要具备SpringMVC源码功底 推荐学习https://www.cnblog ...

  10. Vue2.0源码学习(3) - 组件的创建和patch过程

    组件化 组件化是vue的另一个核心思想,所谓的组件化就,就是说把页面拆分成多个组件(component),每个组件依赖的css.js.图片等资源放在一起开发和维护.组件是资源独立的,在内部系统中是可以 ...

随机推荐

  1. 云图说|初识华为云数据库GaussDB(for openGauss)

    摘要:本文带你了解华为云华为云数据库GaussDB(for openGauss),将AI 技术融入分布式数据库的全生命周期,实现自运维.自管理.自调优.故障自诊断和自愈. 本文分享自华为云社区< ...

  2. 解读知识蒸馏模型TinyBert

    摘要:本篇文章的重点在于改进信息瓶颈的优化机制,并且围绕着高纬空间中互信息难以估计,以及信息瓶颈优化机制中的权衡难题这两个点进行讲解. 本文分享自华为云社区<[云驻共创]美文赏析:大佬对变分蒸馏 ...

  3. IOS证书制作教程

    ​ 转载:IOS证书制作教程 点击苹果证书 按钮 ​ 编辑 点击新增 ​ 编辑 输入证书密码,名称 这个密码不是账号密码,而是一个保护证书的密码,是p12文件的密码,此密码设置后没有其他地方可以找到, ...

  4. matplotlib 图表生成

    条形颜色演示 import matplotlib.pyplot as plt ''' 将plt.subplots()函数的返回值赋值给fig和ax俩个变量 plt.subplots()是一个函数,返回 ...

  5. gunicorn 高性能wsgi服务器

    参考: https://zhuanlan.zhihu.com/p/102716258 Gunicorn是什么 Gunicorn Green Unicorn 是一个 UNIX 下的 WSGI HTTP ...

  6. mybatis使用oracle进行添加数据的心得

    本次博主主要进行oralce数据库开发,好久不用oracle,有很多知识点也忘的差不多了,本次主要是复习一下工作中主要使用的一些sql语句编写: 查询 查询语句都是正常的,但是需要注意的是oracle ...

  7. QE01/QA11/QA02屏幕增强

    1.业务需求 需要对来料检验增加"合格数量"和"不合格数量"字段,涉及三个增强开发 2.QE01\QE02\QE03\QE51N屏幕增强 增强表 增强点BADI ...

  8. C#9.0:Top-Level Programs

    我们称之为顶级层序 用 C# 编写一个简单的程序需要大量的样板代码,引用,类.方法.结构体等: 1 class Program 2 { 3 static void Main(string[] args ...

  9. Codeforces Round #741 (Div. 2) 个人题解 A~D

    比赛链接:Here 1562A. The Miracle and the Sleeper 题意: 给出 \(l,r\) 求出最大化的 \(a\ mod\ b\) (\(l\le b\le b\le a ...

  10. 2D 可视赋能智慧水务绿色集约化发展

    前言 随着国家对环境保护治理程度的日益重视,各地政府积极响应国家政策,在共同聚焦生态文明建设下,急速催生了水务行业数字化转型.如今 "供排污"一体化管理系统成为行业发展的重要趋势, ...