Spring 5.0源码解读:Bean XML配置文件解析类XmlBeanDefinitionReader

引言


XmlBeanDefinitionReader是Spring中用来将Bean的XML配置文件转换为多个BeanDefinition对象的工具类,一个BeanDefinition对象对应一个标签中的信息。这样,BeanFactory或者ApplicationContext仅需要负责根据BeanDefinition的内容来生成的bean的实例。


测试代码


我们以下列代码为例分析XmlBeanDefinitionReader解析XML文档的过程:

@Test
public void beanFactoryTest() {
	Resource xmlSrc = new ClassPathResource("test/context.xml");
	BeanFactory bf = new DefaultListableBeanFactory();
	BeanDefinitionReader bdr = new XmlBeanDefinitionReader((BeanDefinitionRegistry) bf);
	bdr.loadBeanDefinitions(xmlSrc);
	TestBean tb = bf.getBean("testBean", TestBean.class);
	System.out.println(tb);
}

上述代码执行步骤如下:
1、通过ClassPathResource对象将classpath目录下的Bean XML配置文件封装起来。
在Spring框架中,对其内部使用到的资源通过一个抽象接口Resource表示。这些资源可以来自于文件系统、网络、内存(字节数组)等。
Resource类图:
Spring 5.0源码解读:Bean XML配置文件解析类XmlBeanDefinitionReader_第1张图片
2、构造一个BeanFactory对象DefaultListableBeanFactory
3、构造XmlBeanDefinitionReader,并传入目标BeanFactory对象,表示将Bean配置文件的内容写入到BeanDefinitionRegistry中(DefaultListableBeanFactory实现了BeanDefinitionRegistry)。
4、调用XmlBeanDefinitionReaderloadBeanDefinitions(Resouce)方法将配置文件内容写入BeanFactory。
5、执行获取Bean的业务逻辑…


XmlBeanDefinitionReader实例化过程


UML类图:
Spring 5.0源码解读:Bean XML配置文件解析类XmlBeanDefinitionReader_第2张图片
XmlBeanDefinitionReader实现了两个接口:EnvironmentCapableBeanDefinitionReader接口,实现了EnvironmentCapable接口表示这个类可以存储一些环境有关的变量(Spring会默认存储系统变量),实现了BeanDefinitionReader接口表示这是一个BeanDefinition的读取工具。

BeanDefinitionReader接口定义了一些基本的API:

public interface BeanDefinitionReader {
	//获取负责保存BeannDefinition的对象(在上述例子中也就是XmlBeanDefinitionReader)
	BeanDefinitionRegistry getRegistry();
	//获取资源加载器
	@Nullable ResourceLoader getResourceLoader();
	//获取加载Bean的类加载器,可以为null
	@Nullable ClassLoader getBeanClassLoader();
	//获取Bean名称生成器
	BeanNameGenerator getBeanNameGenerator();
	//读取目标资源文件的方法
	int loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException;
	int loadBeanDefinitions(Resource... resources) throws BeanDefinitionStoreException;
	int loadBeanDefinitions(String location) throws BeanDefinitionStoreException;
	int loadBeanDefinitions(String... locations) throws BeanDefinitionStoreException;
}

了解这些接口后,我们来分析XmlBeanDefinitionReader的构造方法:
XmlBeanDefinitionReader仅提供了一个构造方法,这个构造方法需要传入一个BeanDefinitionRegistry实例:

public XmlBeanDefinitionReader(BeanDefinitionRegistry registry) {
	super(registry);
}

这里调用到了父类AbstractBeanDefinitionReader构造方法:

protected AbstractBeanDefinitionReader(BeanDefinitionRegistry registry) {
	Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
	this.registry = registry;
	//如果这个registry同时也能够充当ResourceLoader的角色,那么赋给成员变量resourceLoader
	if (this.registry instanceof ResourceLoader) {
		this.resourceLoader = (ResourceLoader) this.registry;
	} else { //否则采用PathMatchingResourcePatternResolver作为ResourceLoader
		this.resourceLoader = new PathMatchingResourcePatternResolver();
	}

	// 如果这个registry能够充当EnvironmentCapable的角色,那么赋给成员变量environment
	if (this.registry instanceof EnvironmentCapable) {
		this.environment = ((EnvironmentCapable) this.registry).getEnvironment();
	} else { //否则采用StandardEnvironment作为EnvironmentCapable
		this.environment = new StandardEnvironment();
	}
}

需要注意的是,ApplicationContext接口继承了ResourceLoader这个接口,而对于非ApplicationContextBeanFactory实现类(例如DefaultListableBeanFactory)则没有实现这个接口。同样对于EnvironmentCapable来说,DefaultListableBeanFactory也没有实现这个接口,但是ApplicationContext接口继承了EnvironmentCapable接口。

所以,如果传入的registry是DefaultListableBeanFactory实例,那么this.registry instanceof ResourceLoader会返回false,this.registry instanceof EnvironmentCapable同样也是返回false。

EnvironmentCapable接口的作用刚才也提到过了,这里说一下它的实现类StandardEnvironmentStandardEnvironment默认存储了系统变量(内部通过System类的静态方法getenv()getProperties()获取),有兴趣的可以自己去看看源码,比较简单,这里就不罗列了。

这里提一下ResourceLoader这个接口。从接口名字上可以看出,这个接口应该是定义了载入资源的方法,事实上也确实如此:

public interface ResourceLoader {
	String CLASSPATH_URL_PREFIX = ResourceUtils.CLASSPATH_URL_PREFIX;
	Resource getResource(String location);
	@Nullable ClassLoader getClassLoader();
}

ResourceLoader接口的实现类PathMatchingResourcePatternResolver采用了装饰者设计模式,其内部是通过DefaultResourceLoader实现的,那么DefaultResourceLoader是怎样实现上面这两个方法的呢?

@Override
public Resource getResource(String location) {
	Assert.notNull(location, "Location must not be null");
	//首先根据持有的ProtocolResolver解析location
	for (ProtocolResolver protocolResolver : this.protocolResolvers) {
		Resource resource = protocolResolver.resolve(location, this);
		if (resource != null) {
			return resource;
		}
	}
	//如果location以"/"为开头,那么则认定location为文件系统的绝对路径,所以从文件系统中查找
	if (location.startsWith("/")) {
		return getResourceByPath(location);
	} else if (location.startsWith(CLASSPATH_URL_PREFIX)) { //如果以classpath:开头,那么从classpath中查找
		return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length()), getClassLoader());
	} else {
		try {
			// 尝试通过URL查找
			URL url = new URL(location);
			return (ResourceUtils.isFileURL(url) ? new FileUrlResource(url) : new UrlResource(url));
		}
		catch (MalformedURLException ex) { //如果URL获取失败,那么尝试从文件系统中查找
			return getResourceByPath(location);
		}
	}
}

ProtocolResolver 接口定义了根据路径字符串获取资源的方法,若这个ProtocolResolver 无法获取到资源,那么应当返回null。用户可以调用addProtocolResolver(ProtocolResolver resolver)方法来添加自定义的ProtocolResolver

Bean XML文档的载入


构造完XmlBeanDefinitionReader后,接下来的工作就是调用它的loadBeanDefinitions(Resource resource)方法进行文档的解析了。

@Override
public int loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException {
	return loadBeanDefinitions(new EncodedResource(resource));
}

public int loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException {
	Assert.notNull(encodedResource, "EncodedResource must not be null");
	if (logger.isTraceEnabled())
		logger.trace("Loading XML bean definitions from " + encodedResource);
	//获取当前线程正在加载的资源
	Set<EncodedResource> currentResources = this.resourcesCurrentlyBeingLoaded.get();
	//如果没有加载过的资源,那么初始化一个HashSet用来存放当前线程正在加载的资源
	if (currentResources == null) { 
		currentResources = new HashSet<>(4);
		this.resourcesCurrentlyBeingLoaded.set(currentResources);
	}
	//将这个资源添加到Set集合中,若添加失败(Set集合存在这个Resource),则说明发生了配置文件的循环依赖,抛出异常
	if (!currentResources.add(encodedResource)) { 
		throw new BeanDefinitionStoreException("Detected cyclic loading of " + 
							encodedResource + " - check your import definitions!");
	}
	try { //将资源转换为InputSource
		InputStream inputStream = encodedResource.getResource().getInputStream();
		try {
			InputSource inputSource = new InputSource(inputStream);
			if (encodedResource.getEncoding() != null) {
				inputSource.setEncoding(encodedResource.getEncoding());
			}
			//开始进行正式的解析工作
			return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
		} finally {
			inputStream.close();
		}
	} catch (IOException ex) {
		throw new BeanDefinitionStoreException(
					"IOException parsing XML document from " + encodedResource.getResource(), ex);
	} finally {
		//配置文件解析完成后,从Set集合中移除
		currentResources.remove(encodedResource);
		if (currentResources.isEmpty()) //如果当前线程没有正在加载的资源了,则移除这个Set集合
			this.resourcesCurrentlyBeingLoaded.remove();
	}
}

loadBeanDefinitions(Resource resource)方法执行流程如下:
1、将Resource转换为EncodedResource,表示这是一个需要进行解码的资源(因为配置文件属于文本文件,需要对字符进行解码来防止乱码)。
2、根据一个NamedThreadLocalThreadLocal的子类,仅添加了一个name成员变量,用来表示名称)对象来获取当前线程正在加载的配置文件(存放在一个Set集合中)。如果没有获取到这个集合,则说明当前线程没有任何配置文件处于加载过程,此时会实例化一个HashSet并通过NamedThreadLocal绑定到当前线程,并将当前传入的配置文件EncodedResource添加到Set集合。如果获取到了Set集合并且发现这个集合已经存在这个EncodedResource对象了(EncodedResource重写了equals方法,通过资源文件路径来判定是否相等),则说明配置文件发生了循环依赖,抛出BeanDefinitionStoreException异常。

关于ThreadLocal的原理,可以参考我的博客:https://blog.csdn.net/abc123lzf/article/details/81978210。

那么什么时候会发生配置文件的循环依赖?例如这里有两个Bean的XML配置文件:
context1.xml

<beans>
	<import resource="context2.xml" />
beans>

context2.xml

<beans>
	<import resource="context1.xml" />
beans>

这两个配置文件互相依赖对方,在加载过程中就会抛出异常。

3、将EncodedResource转换为InputSource对象(InputSource不是Spring自带的,属于包org.xml.sax),然后,调用doLoadBeanDefinitions进行正式的配置文件解析工作

4、解析完成后,将EncodedResource从Set集合中删除,表示这个配置文件已经解析完成。

现在我们来看doLoadBeanDefinitions做了什么工作:

protected int doLoadBeanDefinitions(InputSource inputSource, Resource resource)
			throws BeanDefinitionStoreException {
	try {
		Document doc = doLoadDocument(inputSource, resource);
		int count = registerBeanDefinitions(doc, resource);
		if (logger.isDebugEnabled()) {
			logger.debug("Loaded " + count + " bean definitions from " + resource);
		}
		return count;
	} catch (Exception ex) {
		//省略异常处理
	}
}

doLoadBeanDefinitions方法有两个步骤:
1、调用doLoadDocument方法将配置文件内容转换为org.w3c.dom.Document对象,如果配置文件有任何XML语法错误,那么会在这里抛出异常。
2、调用registerBeanDefinitions方法,解析Document对象中的节点内容,将其转换为BeanDefinition对象然后注册到BeanDefinitionRegistry上(也就是测试代码中的DefaultListableBeanFactory),返回注册的BeanDefinition对象数量。

doLoadDocument方法更多的是涉及到XML语法的解析工作了,所以这里就不介绍其中的原理了。
来看registerBeanDefinitions方法:

public int registerBeanDefinitions(Document doc, Resource resource) throws BeanDefinitionStoreException {
	//实例化DefaultBeanDefinitionDocumentReader
	BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();
	//获取BeanDefinitionRegistry持有的BeanDefinition数量
	int countBefore = getRegistry().getBeanDefinitionCount();
	//解析配置,并将新解析到的BeanDefinition注册到BeanDefinitionRegistry
	documentReader.registerBeanDefinitions(doc, createReaderContext(resource));
	//返回本次解析的BeanDefinition数量
	return getRegistry().getBeanDefinitionCount() - countBefore;
}

我们来解读一下上述方法的执行细节:
1、首先调用createBeanDefinitionDocumentReader方法构造一个BeanDefinitionDocumentReader

protected BeanDefinitionDocumentReader createBeanDefinitionDocumentReader() {
	return BeanUtils.instantiateClass(this.documentReaderClass);
}

这里直接通过反射构造了DefaultBeanDefinitionDocumentReader对象。
BeanDefinitionDocumentReader接口定义了以下方法:

void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) 
		throws BeanDefinitionStoreException;

这个方法的职责是将Document节点的内容读入到XmlReaderContext

2、获取BeanDefinitionRegistry(即DefaultListableBeanFactory)持有的BeanDefinition数量,这里涉及到BeanFactory源码部分,本文就不说明了。

3、调用createReaderContext方法创建XmlReaderContext:

public XmlReaderContext createReaderContext(Resource resource) {
	return new XmlReaderContext(resource, this.problemReporter, this.eventListener,
			this.sourceExtractor, this, getNamespaceHandlerResolver());
}

XmlReaderContext这个类的名字上来看,应当是在文档解析过程中提供全局信息的一个类。构造XmlReaderContext对象一共需要传入6个变量,分别是:
Resource实例:代表Bean XML配置文件的资源对象
ProblemReporter实例:用来记录错误信息,这里传入的是FailFastProblemReporter,它通过日志的方式记录错误信息。
ReaderEventListener实例:提供一个读入事件监听服务,但是这里传入的是EmptyReaderEventListener,查看源码发现方法实现均为空。
XmlBeanDefinitionReader实例:也就是this
NamespaceHandlerResolver实例:根据XML文档的Namespace来提供不同的NamespaceHandler,用来应对XML文档中的自定义标签,例如

创建完XmlReaderContext后,马上就会执行DefaultBeanDefinitionDocumentReaderregisterBeanDefinitions方法进行正式的解析工作。

4、解析完成后,返回本次读入的BeanDefinition数量。


Bean XML文档的解析


下面我们从刚才的第三步继续深究:

@Override
public void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {
	this.readerContext = readerContext; //首先绑定ReaderContext
	doRegisterBeanDefinitions(doc.getDocumentElement()); //提取出ROOT标签,即
}

这里会获取XML配置根标签,然后调用doRegisterBeanDefinitions方法:

protected void doRegisterBeanDefinitions(Element root) {
	BeanDefinitionParserDelegate parent = this.delegate;
	this.delegate = createDelegate(getReaderContext(), root, parent);
	if (this.delegate.isDefaultNamespace(root)) { //如果根标签是默认标签
		//处理标签的profile属性
		String profileSpec = root.getAttribute(PROFILE_ATTRIBUTE);
		if (StringUtils.hasText(profileSpec)) {
			String[] specifiedProfiles = StringUtils.tokenizeToStringArray(
					profileSpec, BeanDefinitionParserDelegate.MULTI_VALUE_ATTRIBUTE_DELIMITERS);
			if (!getReaderContext().getEnvironment().acceptsProfiles(specifiedProfiles)) {
				if (logger.isDebugEnabled()) {
					logger.debug("Skipped XML bean definition file due to specified profiles [" + profileSpec +
							"] not matching: " + getReaderContext().getResource());
				}
				return;
			}
		}
	}
	preProcessXml(root); //解析前的处理,默认实现为空,提供给子类重写
	parseBeanDefinitions(root, this.delegate); //解析中的元素
	postProcessXml(root); //解析后的处理,默认实现为空,提供给子类重写
	this.delegate = parent;
}

该方法首先会检查当前环境是否和配置文件相符,即获取标签的profile属性,然后通过调用EnvironmentacceptsProfiles方法进行判定。若不相符,则放弃解析当前配置文件。关于标签的profile属性,可以参考这篇博客。

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);
	}
}

上述代码对不同类型标签有着不同的解析方式。我们先来看根标签为默认标签、子标签也为默认标签的解析方式:

1、子标签为默认标签的解析
private void parseDefaultElement(Element ele, BeanDefinitionParserDelegate delegate) {
	if (delegate.nodeNameEquals(ele, IMPORT_ELEMENT)) { //解析标签
		importBeanDefinitionResource(ele);
	} else if (delegate.nodeNameEquals(ele, ALIAS_ELEMENT)) { //解析标签
		processAliasRegistration(ele);
	} else if (delegate.nodeNameEquals(ele, BEAN_ELEMENT)) { //解析标签
		processBeanDefinition(ele, delegate);
	} else if (delegate.nodeNameEquals(ele, NESTED_BEANS_ELEMENT)) { //解析标签
		doRegisterBeanDefinitions(ele);
	}
}

1.1 标签解析

protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {
	//解析默认标签
	BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
	if (bdHolder != null) {
		//解析默认标签中的自定义标签
		bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);
		try {
			// Register the final decorated instance.
			BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());
		}
			catch (BeanDefinitionStoreException ex) {
				getReaderContext().error("Failed to register bean definition with name '" +
						bdHolder.getBeanName() + "'", ele, ex);
			}
			// Send registration event.
			getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));
		}
	}

BeanDefinitionParserDelegate是XML配置文件中标签解析的核心类,processBeanDefinition调用了它的parseBeanDefinitionElement方法并传入Element元素,目的是将这个元素转换为BeanDefinitionHolder对象,BeanDefinitionHolder并非BeanDefinition接口的实现类,它的作用是作为一个元素解析过程中的临时信息承载体,它的成员变量包含了Bean的ID、name和BeanDefinition对象:

@Nullable
public BeanDefinitionHolder parseBeanDefinitionElement(Element ele) {
	return parseBeanDefinitionElement(ele, null);
}

@Nullable
public BeanDefinitionHolder parseBeanDefinitionElement(Element ele, @Nullable BeanDefinition containingBean) {
	String id = ele.getAttribute(ID_ATTRIBUTE);			//获取bean的id属性
	String nameAttr = ele.getAttribute(NAME_ATTRIBUTE);	//获取bean的name属性
	//建立一个List集合,用于存放bean的名称
	List<String> aliases = new ArrayList<>();
	if (StringUtils.hasLength(nameAttr)) { //
		//将name属性内容根据分隔符','或';'将其分成多个字符串,并添加到name集合中
		String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr, MULTI_VALUE_ATTRIBUTE_DELIMITERS);
		aliases.addAll(Arrays.asList(nameArr));
	}
	String beanName = id;
	//如果id为空且name不为空
	if (!StringUtils.hasText(beanName) && !aliases.isEmpty()) {
		beanName = aliases.remove(0); //将第一个name作为beanName
		if (logger.isTraceEnabled()) {
			logger.trace("No XML 'id' specified - using '" + beanName +
					"' as bean name and " + aliases + " as aliases");
		}
	}

	if (containingBean == null) { 
		//检查这个beanName是否在配置中是独一无二的,如果不是就会发生错误
		checkNameUniqueness(beanName, aliases, ele);
	}
	//创建Bean信息的承载实例AbstractBeanDefinition(BeanDefinition的抽象子类)
	AbstractBeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName, containingBean);
	if (beanDefinition != null) {
		if (!StringUtils.hasText(beanName)) { //如果beanName为空
			try { 
				if (containingBean != null) { //生成一个内部使用的beanName
					beanName = BeanDefinitionReaderUtils.generateBeanName( 
							beanDefinition, this.readerContext.getRegistry(), true);
				} else {
					beanName = this.readerContext.generateBeanName(beanDefinition);
					String beanClassName = beanDefinition.getBeanClassName();
					if (beanClassName != null && beanName.startsWith(beanClassName) && 
							beanName.length() > beanClassName.length() &&
							!this.readerContext.getRegistry().isBeanNameInUse(beanClassName)) {
						aliases.add(beanClassName);
					}
				}
				if (logger.isTraceEnabled()) {
					logger.trace("Neither XML 'id' nor 'name' specified - " +
							"using generated bean name [" + beanName + "]");
				}
			} catch (Exception ex) {
				error(ex.getMessage(), ele);
				return null;
			}
		}
		String[] aliasesArray = StringUtils.toStringArray(aliases);
		//将解析后的信息封装到BeanDefinitionHolder
		return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);
	}
	return null;
}

现在来分析一下parseBeanDefinitionElement方法的执行流程:
1、提取Bean的id、name属性,如果没有设置id,就取name作为它的id。
2、解析文档内容并封装为一个AbstractBeanDefinition对象
3、如果id和name属性都为空,并且containingBean也为null,那么就调用BeanDefinitionReaderUtils工具类的静态方法generateBeanName生成一个内部使用的id,这个id一般为Bean的全限定类名,如果这个bean是一个内部Bean,那么则名称为父Bean的名称+"$child"+"#"+这个BeanDefinition的内存地址(通过System类的identityHashCode获得)。
4、最后将Bean的解析结果封装为BeanDefinitionHolder返回。

下面是第2步AbstractBeanDefinition的生成过程。

/* File: org.springframework.beans.factory.xml.BeanDefinitionParserDelegate */
@Nullable
public AbstractBeanDefinition parseBeanDefinitionElement(Element ele, String beanName,
		@Nullable BeanDefinition containingBean) {
	this.parseState.push(new BeanEntry(beanName));

	String className = null;
	if (ele.hasAttribute(CLASS_ATTRIBUTE)) { //解析class属性
		className = ele.getAttribute(CLASS_ATTRIBUTE).trim();
	}
	String parent = null;
	if (ele.hasAttribute(PARENT_ATTRIBUTE)) { //解析parent属性
		parent = ele.getAttribute(PARENT_ATTRIBUTE);
	}
	try {
		//根据class名称和父Bean构造BeanDefinition
		AbstractBeanDefinition bd = createBeanDefinition(className, parent);
		//解析bean标签下的属性(非子标签)
		parseBeanDefinitionAttributes(ele, beanName, containingBean, bd);
		//解析标签
		bd.setDescription(DomUtils.getChildElementValueByTagName(ele, DESCRIPTION_ELEMENT));
		//解析标签
		parseMetaElements(ele, bd);
		//解析标签
		parseLookupOverrideSubElements(ele, bd.getMethodOverrides());	
		//解析标签
		parseReplacedMethodSubElements(ele, bd.getMethodOverrides());
		//解析标签
		parseConstructorArgElements(ele, bd);
		//解析标签
		parsePropertyElements(ele, bd);
		//解析标签
		parseQualifierElements(ele, bd);
		//将XML配置文件的Resource对象赋给BeanDefinition
		bd.setResource(this.readerContext.getResource());
		bd.setSource(extractSource(ele));

		return bd;
	} catch (ClassNotFoundException ex) {
		error("Bean class [" + className + "] not found", ele, ex);
	} catch (NoClassDefFoundError err) {
		error("Class that bean class [" + className + "] depends on not found", ele, err);
	} catch (Throwable ex) {
		error("Unexpected failure during bean definition parsing", ele, ex);
	} finally {
		this.parseState.pop();
	}
	return null;
}

上述方法主要是用来处理标签的属性及其包含的子标签。
我们先来简单回顾一下有哪些属性和子标签:
1、标签的属性
parent:用来指定父Bean,这里应当写入父Bean的名称,可以为抽象Bean
scope:Bean的作用域,Spring内置的有singleton(单例)、prototype(多例)、request(请求范围)、session(会话)、global-session(全局)
abstract:是否为抽象Bean,抽象Bean一般是提供一个Bean的模板,不包含class信息。其值只可为true或者false。
lazy-init:是否采用延迟初始化策略,对于非ApplicationContextBeanFactory实现类来说,是默认开启延迟初始化策略的,只有在需要获取Bean的实例时候才会加载这个类,而ApplicationContext是默认关闭延迟初始化的,即在容器加载过程中完成类的加载操作,后续仅需要进行实例化即可。其值只可为true、false或者default。
autowire:即自动装配机制,当别的Bean需要依赖这个Bean的时候,是通过什么样的策略来识别这个Bean并注入到别的Bean。其值只可为byNamebyTypeconstructordefaultno。具体可参照这篇博客:https://www.cnblogs.com/ViviChan/p/4981539.html
depends-on:表示依赖于指定的Bean,在当前Bean初始化的时候,会优先初始化depends-on中指定的Bean
autowire-candidate:Bean的候选属性,同样可以参照上面的博客。其值只可为true、false或者default。
primary:表示是否是优先的Bean,如果容器中有多个同一类型的Bean,如果不指定一个primary为true的Bean,那么在注入时就会因为存在多个类型匹配的Bean而抛出异常。当指定了一个primary为true的Bean后,就不会因此而抛出异常。其值只可为true、false
init-method:Bean的默认初始化方法
destory-method:Bean的默认销毁方法
factory-bean:指向实例工厂方法的bean
factory-method:指向实例工厂方法的名字

2、子标签
:Bean的描述信息
:用来存储一些键值型的参数,通过调用BeanDefinitiongetAttribute方法获得
:用来通过方法来实现依赖注入,可以参考博客:https://www.cnblogs.com/ViviChan/p/4981619.html
:用来替换该Bean存在的方法,其属性有namereplacer。可以参考这篇博客
:通过成员变量注入
:通过构造方法注入
:指定Bean的一个特殊名称,防止在注入时其它Bean时因为有多个匹配的Bean而发生异常。

了解完了后,我们来看看AbstractBeanDefinition是如何构造的:

protected AbstractBeanDefinition createBeanDefinition(@Nullable String className, @Nullable String parentName)
		throws ClassNotFoundException {
	return BeanDefinitionReaderUtils.createBeanDefinition(parentName, className, 
				this.readerContext.getBeanClassLoader());
}

public static AbstractBeanDefinition createBeanDefinition(@Nullable String parentName, 
		@Nullable String className, @Nullable ClassLoader classLoader) throws ClassNotFoundException {
	GenericBeanDefinition bd = new GenericBeanDefinition();
	bd.setParentName(parentName); //设置父Bean的名称
	if (className != null) {
		if (classLoader != null) { //如果设置了类加载器,那么直接初始化该类
			bd.setBeanClass(ClassUtils.forName(className, classLoader));
		} else { //否则,只是记录这个类的类名
			bd.setBeanClassName(className);
		}
	}
	return bd;
}

这里的类加载器默认从ReaderContext获取,而ReaderContext内部则默认通过XmlBeanDefinitionReader获得。XmlBeanDefinitionReader默认的Bean类加载器为null,表示进行延迟初始化。所以在这个地方构造了GenericBeanDefinition后,仅仅只是记录了类名而不进行类的初始化。除非调用了XmlBeanDefinitionReadersetBeanClassLoader方法强制指定了类加载器。

GenericBeanDefinition继承了AbstractBeanDefinition,仅添加了一个成员变量parentName,表示父Bean的名称。
Spring 5.0源码解读:Bean XML配置文件解析类XmlBeanDefinitionReader_第3张图片

构造完GenericBeanDefinition后,就会开始进行元素的解析工作,即parseBeanDefinitionAttributes方法,这段代码限于篇幅就不贴了,其实很简单,就是调用了AbstractBeanDefinition一些set方法设置每个属性的值,感兴趣的朋友可以自己去看一看。

随后的操作步骤:
1、根据内容将描述信息设置到GenericBeanDefinition
2、根据内容设置参数信息到GenericBeanDefinition

public void parseMetaElements(Element ele, BeanMetadataAttributeAccessor attributeAccessor) {
	NodeList nl = ele.getChildNodes();
	for (int i = 0; i < nl.getLength(); i++) {
		Node node = nl.item(i);
		if (isCandidateElement(node) && nodeNameEquals(node, META_ELEMENT)) {
			Element metaElement = (Element) node;
			String key = metaElement.getAttribute(KEY_ATTRIBUTE);
			String value = metaElement.getAttribute(VALUE_ATTRIBUTE);
			BeanMetadataAttribute attribute = new BeanMetadataAttribute(key, value);
			attribute.setSource(extractSource(metaElement));
			attributeAccessor.addMetadataAttribute(attribute);
		}
	}
}

parseMetaElements会将每个元素中的key和value的值封装为一个BeanMetadataAttribute对象然后添加到GenericBeanDefinition中。

3、根据标签将其中的name、bean属性封装为一个LookupOverride对象并添加到GenericBeanDefinition持有的MethodOverrides对象中。逻辑和上述代码类似。
4、根据标签将其中的内容封装为ReplaceOverride对象并添加到GenericBeanDefinition持有的MethodOverrides对象中。
5、将每个内容封装为ConstructorArgumentValues.ValueHolder对象并添加到GenericBeanDefinition持有的ConstructorArgumentValues中。
6、将每个标签封装为一个PropertyValue对象然后添加到GenericBeanDefinition
7、将标签封装为一个BeanMetadataAttribute对象然后添加到一个新构造的AutowireCandidateQualifier中,然后将AutowireCandidateQualifier添加到GenericBeanDefinition

回过头来继续分析processBeanDefinition方法,当parseBeanDefinitionElement返回一个BeanDefinitionHolder实例后,会继续调用BeanDefinitionParserDelegatedecorateBeanDefinitionIfRequired方法,这个方法是用来解析自定义标签的:

public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder) {
	return decorateBeanDefinitionIfRequired(ele, definitionHolder, null);
}

public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, 
		BeanDefinitionHolder definitionHolder, @Nullable BeanDefinition containingBd) {
	BeanDefinitionHolder finalDefinition = definitionHolder;
	//获取这个标签下所有的自定义属性
	NamedNodeMap attributes = ele.getAttributes();
	for (int i = 0; i < attributes.getLength(); i++) {
		Node node = attributes.item(i);
		finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);
	}
	//获取这个标签下所有的自定义标签
	NodeList children = ele.getChildNodes();
	for (int i = 0; i < children.getLength(); i++) {
		Node node = children.item(i);
		if (node.getNodeType() == Node.ELEMENT_NODE) {
			finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);
		}
	}
	return finalDefinition;
}

上述方法的主要功能就是获取这个标签下所有的自定义标签及其属性,获取到Node对象后,就会调用decorateIfRequired方法进行进一步的解析工作:

public BeanDefinitionHolder decorateIfRequired(Node node, BeanDefinitionHolder originalDef, 
		@Nullable BeanDefinition containingBd) {
	String namespaceUri = getNamespaceURI(node); //获取这个Node的Namespace
	//如果Namespace不为null且不是默认的namespace
	if (namespaceUri != null && !isDefaultNamespace(namespaceUri)) {
		//获取对应的NamespaceHandler进行处理
		NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
		if (handler != null) { //如果找到了对应的NamespaceHandler,那么就调用它的decorate方法解析
			BeanDefinitionHolder decorated = handler.decorate(node, originalDef, 
				new ParserContext(this.readerContext, this, containingBd));
			if (decorated != null) { //如果解析成功,就返回
				return decorated;
			}
		//如果是Spring的Namespace却没有找到对应的NamespaceHandler,那么抛出异常
		} else if (namespaceUri.startsWith("http://www.springframework.org/")) { 
			error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", node);
		} else { //如果没有找到NamespaceHandler
			if (logger.isDebugEnabled()) {
				logger.debug("No Spring NamespaceHandler found for XML schema namespace [" + namespaceUri + "]");
			}
		}
	}
	//如果执行到这里,则表示并没有对这个Node进行任何的解析工作
	return originalDef;
}

NamespaceHandler接口定义了解析自定义标签的方法,用户如果想实现自己的自定义标签并成功地载入容器,那么就需要实现这个接口。

现在,这个标签中所有的元素和标签都已经解析完成,那么剩下的工作就是将这个BeanDefinitionHolder中的BeanDefinition注册到BeanDefinitionRegistry上(也就是测试代码中的DefaultListableBeanFactory),这里是通过工具类BeanDefinitionReaderUtils的静态方法registerBeanDefinition完成的:

public static void registerBeanDefinition(BaeanDefinitionHolder definitionHolder,
		 BeanDefinitionRegistry registry) throws BeanDefinitionStoreException {
	String beanName = definitionHolder.getBeanName(); //获取bean的名称
	//注册到BeanDefinitionRegistry
	registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition());
	//注册id和name的对应关系
	String[] aliases = definitionHolder.getAliases();
	if (aliases != null) {
		for (String alias : aliases) {
			registry.registerAlias(beanName, alias);
		}
	}
}

1.2 标签解析
在当前配置文件中,标签用来引入别的配置文件,其属性resource制定了其路径。

protected void importBeanDefinitionResource(Element ele) {
	//获取resource属性
	String location = ele.getAttribute(RESOURCE_ATTRIBUTE);
	if (!StringUtils.hasText(location)) {
		getReaderContext().error("Resource location must not be empty", ele);
		return;
	}
	//解析路径中的系统属性
	location = getReaderContext().getEnvironment().resolveRequiredPlaceholders(location);
	Set<Resource> actualResources = new LinkedHashSet<>(4);
	boolean absoluteLocation = false;
	//判断路径是绝对路径还是相对路径
	try {
		absoluteLocation = ResourcePatternUtils.isUrl(location) || ResourceUtils.toURI(location).isAbsolute();
	} catch (URISyntaxException ex) {
	}
	if (absoluteLocation) {	//如果是绝对路径
		try { //加载这个资源
			int importCount = getReaderContext().getReader().loadBeanDefinitions(location, actualResources);
			if (logger.isTraceEnabled()) {
				logger.trace("Imported " + importCount + " bean definitions from URL location [" + location + "]");
			}
		} catch (BeanDefinitionStoreException ex) {
			getReaderContext().error("Failed to import bean definitions from URL location [" + location + "]", ele, ex);
		}
	} else { //如果是相对路径
		try {
			int importCount;
			Resource relativeResource = getReaderContext().getResource().createRelative(location);
			if (relativeResource.exists()) { //如果资源存在,则加载这个资源
				importCount = getReaderContext().getReader().loadBeanDefinitions(relativeResource);
				actualResources.add(relativeResource);
			} else { //如果不存在,则转换为URL处理
				String baseLocation = getReaderContext().getResource().getURL().toString();
				importCount = getReaderContext().getReader().loadBeanDefinitions(
						StringUtils.applyRelativePath(baseLocation, location), actualResources);
			}
			if (logger.isTraceEnabled()) {
				logger.trace("Imported " + importCount + " bean definitions from relative location [" + location + "]");
			}
		} catch (IOException ex) {
			getReaderContext().error("Failed to resolve current resource location", ele, ex);
		} catch (BeanDefinitionStoreException ex) {
			getReaderContext().error(
						"Failed to import bean definitions from relative location [" + location + "]", ele, ex);
		}
	}
	Resource[] actResArray = actualResources.toArray(new Resource[0]);
	//通知事件监听器,在本文案例中什么也不会做
	getReaderContext().fireImportProcessed(location, actResArray, extractSource(ele));
}

上述方法的大致流程为:
1、获取resource属性的值
2、转换为路径信息
3、递归调用XmlBeanDefinitionReader的解析方法
4、通知ReaderContext绑定的事件监听器

1.3 标签解析
标签的作用的是给Bean指定别名,和标签中的name属性类似。

protected void processAliasRegistration(Element ele) {
	String name = ele.getAttribute(NAME_ATTRIBUTE); //获取name属性
	String alias = ele.getAttribute(ALIAS_ATTRIBUTE); //获取alias属性
	boolean valid = true;
	if (!StringUtils.hasText(name)) {
		getReaderContext().error("Name must not be empty", ele);
		valid = false;
	}
	if (!StringUtils.hasText(alias)) {
		getReaderContext().error("Alias must not be empty", ele);
		valid = false;
	}
	if (valid) {
		try {
			//向BeanDefinitionRegistry注册name与别名的映射关系
			getReaderContext().getRegistry().registerAlias(name, alias);
		} catch (Exception ex) {
			getReaderContext().error("Failed to register alias '" + alias +
				"' for bean with name '" + name + "'", ele, ex);
		}
		//通知事件监听器
		getReaderContext().fireAliasRegistered(name, alias, extractSource(ele));
	}
}

1.4 标签解析
XML配置文件支持标签的双重嵌套,其作用和类似,其解析过程和顶层的标签没有区别,都是调用了DefaultBeanDefinitionDocumentReaderdoRegisterBeanDefinitions方法。

2、 子标签为自定义标签的解析
若在遍历标签中发现了这个标签为自定义标签时,便会调用BeanDefinitionParserDelegateparseCustomElement方法:

@Nullable
public BeanDefinition parseCustomElement(Element ele) {
	return parseCustomElement(ele, null);
}

@Nullable
public BeanDefinition parseCustomElement(Element ele, @Nullable BeanDefinition containingBd) {
	String namespaceUri = getNamespaceURI(ele);
	if (namespaceUri == null)
		return null;
	NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
	if (handler == null) {
		error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", ele);
		return null;
	}
	return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));
}

和上面解析的自定义属性和自定义标签类似,都是先获取这个自定义标签的Namespace然后找到其对应的NamespaceHandler进行处理。

你可能感兴趣的:(Spring 5.0源码解读:Bean XML配置文件解析类XmlBeanDefinitionReader)