一起学SF框架系列5.7-模块Beans-BeanDefinition解析

开发人员按元数据规则定义了应用bean,了解SF如何根据定义解析成BeanDefiniton有助于深入理解框架实现。解析过程如下:

资源加载

从资源文件加载bean的元数据配置,实际过程如下图:
在这里插入图片描述
实际从指定的XML文件加载bean定义是从XmlBeanDefinitionReader.doLoadBeanDefinitions开始。

解析

解析就是两大步:
1、把资源文件内容通过SAX解析器解析成DOM文档
2、从DOM文档生成BeanDefinition
源代码(XmlBeanDefinitionReader.doLoadBeanDefinitions):

	protected int doLoadBeanDefinitions(InputSource inputSource, Resource resource)throws BeanDefinitionStoreException {try {//将资源解析成DocumentDocument doc = doLoadDocument(inputSource, resource);//基于Document生成BeanDefinitionint count = registerBeanDefinitions(doc, resource);if (logger.isDebugEnabled()) {logger.debug("Loaded " + count + " bean definitions from " + resource);}return count;}catch (BeanDefinitionStoreException ex) {throw ex;}catch (SAXParseException ex) {throw new XmlBeanDefinitionStoreException(resource.getDescription(),"Line " + ex.getLineNumber() + " in XML document from " + resource + " is invalid", ex);}catch (SAXException ex) {throw new XmlBeanDefinitionStoreException(resource.getDescription(),"XML document from " + resource + " is invalid", ex);}catch (ParserConfigurationException ex) {throw new BeanDefinitionStoreException(resource.getDescription(),"Parser configuration exception parsing XML from " + resource, ex);}catch (IOException ex) {throw new BeanDefinitionStoreException(resource.getDescription(),"IOException parsing XML document from " + resource, ex);}catch (Throwable ex) {throw new BeanDefinitionStoreException(resource.getDescription(),"Unexpected exception parsing XML document from " + resource, ex);}}

资源文件到Document

Spring实现比较复杂,基本关系图如下:
在这里插入图片描述
主要是用SAX方式对xml格式元数据解析成DOM文档,同spring本身目的关系不大,不需详细跟踪代码,有兴趣同学可以自己学习了解。

Document到BeanDefinition

主要过程

在这里插入图片描述

XmlBeanDefinitionReader.registerBeanDefinitions(Document doc, Resource resource)

	public int registerBeanDefinitions(Document doc, Resource resource) throws BeanDefinitionStoreException {//生成DOM解析器(负责生成BeanDefinition)BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();//返回已经注册在BeanFactory中的beanDefinitions数量(支持多个元数据配置文件)int countBefore = getRegistry().getBeanDefinitionCount();//生成BeanDefinitiondocumentReader.registerBeanDefinitions(doc, createReaderContext(resource));//返回本次beanDefinitions数量(BeanFactory现有数量-生成前记录的数量)return getRegistry().getBeanDefinitionCount() - countBefore;}

BeanDefinitionDocumentReader.registerBeanDefinitions(Document doc, XmlReaderContext readerContext)

一看就明。

	@Overridepublic void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {this.readerContext = readerContext;doRegisterBeanDefinitions(doc.getDocumentElement());}

BeanDefinitionDocumentReader.doRegisterBeanDefinitions(Element root)

从bean根元素( )开始解析。

	//不理解怎么成deprecation@SuppressWarnings("deprecation")  // for Environment.acceptsProfiles(String...)protected void doRegisterBeanDefinitions(Element root) {//保存BeanDefinitionParser代理BeanDefinitionParserDelegate parent = this.delegate;//生成当前用的BeanDefinitionParser代理,初始化一些默认值this.delegate = createDelegate(getReaderContext(), root, parent);//默认命名空间是"http://www.springframework.org/schema/beans"if (this.delegate.isDefaultNamespace(root)) {//xml文件中沒有使用到"profile",有的话需要加载profile相关值 注1String profileSpec = root.getAttribute(PROFILE_ATTRIBUTE);if (StringUtils.hasText(profileSpec)) {String[] specifiedProfiles = StringUtils.tokenizeToStringArray(profileSpec, BeanDefinitionParserDelegate.MULTI_VALUE_ATTRIBUTE_DELIMITERS);// We cannot use Profiles.of(...) since profile expressions are not supported// in XML config. See SPR-12458 for details.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;}}}//xml预处理(空实现)preProcessXml(root);//生成BeanDefinition,并注册到BeanFactoryparseBeanDefinitions(root, this.delegate);//xml预处理(空实现)postProcessXml(root);//恢复BeanDefinitionParser代理this.delegate = parent;}

注1 什么是profile,参考:一起学SF框架系列6.2-模块core-Environment

BeanDefinitionDocumentReader.parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate)

生成BeanDefinition,并注册到BeanFactory。

	protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {//根节点是默认命名空间(http://www.springframework.org/schema/beans)解析 注1if (delegate.isDefaultNamespace(root)) {//获取根节点下所有子节点NodeList nl = root.getChildNodes();//逐一解析每个子节点for (int i = 0; i < nl.getLength(); i++) {Node node = nl.item(i);if (node instanceof Element ele) {//每个子节点按命名空间对应解析器进行解析if (delegate.isDefaultNamespace(ele)) {parseDefaultElement(ele, delegate);}else {delegate.parseCustomElement(ele);}}}}else {//非默认命名空间解析delegate.parseCustomElement(root);}}

注1:命名空间的理解如下,一个命名空间对应一个解析处理器:
如启用注解配置,需在spring的配置文件applicatinoContext.xml中加入下面这行:
<context:annotation-config />
意思是标签annotation-config属于命名空间context。context就对应一个解析处理器。解析的时候需先获取context对应解析处理器。解析处理器又会初始化一些解析器出來,一个解析器就对应着一個标签。
基于默认命名空间如何解析往下跟踪。

BeanDefinitionDocumentReader.parseDefaultElement(Element ele, BeanDefinitionParserDelegate delegate)

默认命名空间的元素解析。

private void parseDefaultElement(Element ele, BeanDefinitionParserDelegate delegate) {//<import>标签进入这个方法if (delegate.nodeNameEquals(ele, IMPORT_ELEMENT)) {importBeanDefinitionResource(ele);}//<alias>标签进入这个方法else if (delegate.nodeNameEquals(ele, ALIAS_ELEMENT)) {processAliasRegistration(ele);}//<bean>标签进入这个方法else if (delegate.nodeNameEquals(ele, BEAN_ELEMENT)) {//进入这个方法processBeanDefinition(ele, delegate);}//嵌套标签进入这个方法else if (delegate.nodeNameEquals(ele, NESTED_BEANS_ELEMENT)) {// 嵌套标签从doRegisterBeanDefinitions递归开始doRegisterBeanDefinitions(ele);}
}

基于生成bean往下跟踪。

BeanDefinitionDocumentReader.processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate)

把bean元素解析成BeanDefinition并注册到BeanFactory中。

protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {//把元素解析成BeanDefinitionBeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);if (bdHolder != null) {  //装饰beanDefinitionbdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);try {//注册BeanDefinition到工厂中BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());}catch (BeanDefinitionStoreException ex) {getReaderContext().error("Failed to register bean definition with name '" +bdHolder.getBeanName() + "'", ele, ex);}// 发送组件(bean)注册事件getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));}
}

BeanDefinitionParserDelegate.parseBeanDefinitionElement(Element ele)

//把元素解析成BeanDefinition

public BeanDefinitionHolder parseBeanDefinitionElement(Element ele) {return parseBeanDefinitionElement(ele, null);
}
public BeanDefinitionHolder parseBeanDefinitionElement(Element ele, BeanDefinition containingBean) {  //获取元素idString id = ele.getAttribute(ID_ATTRIBUTE);//获取元素nameString nameAttr = ele.getAttribute(NAME_ATTRIBUTE);//记录元素aliasList<String> aliases = new ArrayList<String>();/* id、name、alias关系处理 注1 *///name解析if (StringUtils.hasLength(nameAttr)) {//name属性对应的name值如果有分隔符MULTI_VALUE_ATTRIBUTE_DELIMITERS,那么切分成数组String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr, MULTI_VALUE_ATTRIBUTE_DELIMITERS);//所有name都是别名aliases.addAll(Arrays.asList(nameArr));}//指定了id就用id值作为bean名称String beanName = id;//如果没有id,但是指定了name,就用name值作为bean名称if (!StringUtils.hasText(beanName) && !aliases.isEmpty()) {//拿第一个name值作为bean名称,其余的还是别名beanName = aliases.remove(0);if (logger.isDebugEnabled()) {logger.debug("No XML 'id' specified - using '" + beanName +"' as bean name and " + aliases + " as aliases");}}if (containingBean == null) {        //检查bean名称和别名是否已经被使用了,如果用了就报错checkNameUniqueness(beanName, aliases, ele);}//解析beanDefinition本身AbstractBeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName, containingBean);if (beanDefinition != null) {       //如果没有定义id、name、alias,自动生成beanNameif (!StringUtils.hasText(beanName)) {try {//有containingBeanif (containingBean != null) {beanName = BeanDefinitionReaderUtils.generateBeanName(beanDefinition, this.readerContext.getRegistry(), true);}else {//无containingBean,用容器生成beanNamebeanName = this.readerContext.generateBeanName(beanDefinition);String beanClassName = beanDefinition.getBeanClassName();if (beanClassName != null &&beanName.startsWith(beanClassName) && beanName.length() > beanClassName.length() &&!this.readerContext.getRegistry().isBeanNameInUse(beanClassName)) {//beanClassName同时作为别名aliases.add(beanClassName);}}if (logger.isDebugEnabled()) {logger.debug("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);//组装成完整beanDefinition返回return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);}return null;
}

注1:了解bean的id、name、alias关系,参考:一起学SF框架系列5.1-模块Beans-bean概述

BeanDefinitionParserDelegate.parseBeanDefinitionElement(Element ele, String beanName, @Nullable BeanDefinition containingBean)

不关注bean的名称和别名,只解析BeanDefinition自身。

public AbstractBeanDefinition parseBeanDefinitionElement(Element ele, String beanName, BeanDefinition containingBean) {//解析的beanName进栈(主要考虑元素ele可能还嵌套有子元素,则后续放入子元素,到时子元素先弹出this.parseState.push(new BeanEntry(beanName));String className = null;if (ele.hasAttribute(CLASS_ATTRIBUTE)) {//如果有指定class属性,则拿到class属性值className = ele.getAttribute(CLASS_ATTRIBUTE).trim();}String parent = null;if (ele.hasAttribute(PARENT_ATTRIBUTE)) {           //如果有指定parent属性,则拿到parent属性值parent = ele.getAttribute(PARENT_ATTRIBUTE);}try {       //创建BeanDefinitionAbstractBeanDefinition bd = createBeanDefinition(className, parent);//设置BeanDefinition属性parseBeanDefinitionAttributes(ele, beanName, containingBean, bd);//如果有description子元素,设置到BeanDefinition中bd.setDescription(DomUtils.getChildElementValueByTagName(ele, DESCRIPTION_ELEMENT));//如果有Meta子元素,获取key和value属性,设置到BeanDefinition中parseMetaElements(ele, bd);//如果有lookup-method子元素,获取name和bean属性,设置到BeanDefinition中parseLookupOverrideSubElements(ele, bd.getMethodOverrides());//如果有replaced-method子元素,设置BeanDefinition parseReplacedMethodSubElements(ele, bd.getMethodOverrides());//如果有constructor-arg子元素,设置BeanDefinition的构造方式parseConstructorArgElements(ele, bd);//解析bean的Property的属性设置到BeanDefinition中parsePropertyElements(ele, bd);//子元素qualifier处理parseQualifierElements(ele, bd);//设置定义来源bd.setResource(this.readerContext.getResource());//元素有Sourcebd.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;
}

BeanDefinitionParserDelegate.createBeanDefinition(@Nullable String className, @Nullable String parentName)

创建BeanDefinition并设置属性:parentName和beanClassName

protected AbstractBeanDefinition createBeanDefinition(String className, String parentName)throws ClassNotFoundException {return BeanDefinitionReaderUtils.createBeanDefinition(parentName, className, this.readerContext.getBeanClassLoader());
}/**
* 此方法是BeanDefinitionReaderUtils的静态方法
*/
public static AbstractBeanDefinition createBeanDefinition(String parentName, String className, ClassLoader classLoader) throws ClassNotFoundException {//new通用BeanDefinitionGenericBeanDefinition bd = new GenericBeanDefinition();bd.setParentName(parentName);if (className != null) {if (classLoader != null) {//有classLoader,就导入Classbd.setBeanClass(ClassUtils.forName(className, classLoader));}else {          //无classLoader,只设置BeanClassNamebd.setBeanClassName(className);}}return bd;
}

BeanDefinitionParserDelegate.parseBeanDefinitionAttributes(Element ele, String beanName, @Nullable BeanDefinition containingBean, AbstractBeanDefinition bd)

解析bean定义设置到BeanDefinition属性

public AbstractBeanDefinition parseBeanDefinitionAttributes(Element ele, String beanName,BeanDefinition containingBean, AbstractBeanDefinition bd) {//有singleton属性报错(新的用scope代替)if (ele.hasAttribute(SINGLETON_ATTRIBUTE)) {error("Old 1.x 'singleton' attribute in use - upgrade to 'scope' declaration", ele);}    //如果有scope配置else if (ele.hasAttribute(SCOPE_ATTRIBUTE)) {bd.setScope(ele.getAttribute(SCOPE_ATTRIBUTE));}else if (containingBean != null) {// 按containingBean的Scope设置bd.setScope(containingBean.getScope());}//是否有abstract属性if (ele.hasAttribute(ABSTRACT_ATTRIBUTE)) {bd.setAbstract(TRUE_VALUE.equals(ele.getAttribute(ABSTRACT_ATTRIBUTE)));}//设置lazyInit属性(如果没有设置则为默认值)String lazyInit = ele.getAttribute(LAZY_INIT_ATTRIBUTE);if (DEFAULT_VALUE.equals(lazyInit)) {lazyInit = this.defaults.getLazyInit();}bd.setLazyInit(TRUE_VALUE.equals(lazyInit)); //非TRUE_VALUE则为false//设置autowire属性String autowire = ele.getAttribute(AUTOWIRE_ATTRIBUTE);bd.setAutowireMode(getAutowireMode(autowire));//设置依赖检查属性String dependencyCheck = ele.getAttribute(DEPENDENCY_CHECK_ATTRIBUTE);bd.setDependencyCheck(getDependencyCheck(dependencyCheck));//设置depends-on属性if (ele.hasAttribute(DEPENDS_ON_ATTRIBUTE)) {String dependsOn = ele.getAttribute(DEPENDS_ON_ATTRIBUTE);bd.setDependsOn(StringUtils.tokenizeToStringArray(dependsOn, MULTI_VALUE_ATTRIBUTE_DELIMITERS));}//设置autowire-candidate属性String autowireCandidate = ele.getAttribute(AUTOWIRE_CANDIDATE_ATTRIBUTE);if ("".equals(autowireCandidate) || DEFAULT_VALUE.equals(autowireCandidate)) {String candidatePattern = this.defaults.getAutowireCandidates();if (candidatePattern != null) {String[] patterns = StringUtils.commaDelimitedListToStringArray(candidatePattern);bd.setAutowireCandidate(PatternMatchUtils.simpleMatch(patterns, beanName));}}else {bd.setAutowireCandidate(TRUE_VALUE.equals(autowireCandidate));}//设置primary属性if (ele.hasAttribute(PRIMARY_ATTRIBUTE)) {bd.setPrimary(TRUE_VALUE.equals(ele.getAttribute(PRIMARY_ATTRIBUTE)));}//设置init-method属性if (ele.hasAttribute(INIT_METHOD_ATTRIBUTE)) {String initMethodName = ele.getAttribute(INIT_METHOD_ATTRIBUTE);if (!"".equals(initMethodName)) {bd.setInitMethodName(initMethodName);}}else {//没有init-method属性,设置默认InitMethodif (this.defaults.getInitMethod() != null) {bd.setInitMethodName(this.defaults.getInitMethod());bd.setEnforceInitMethod(false);}}//设置destroy-method属性if (ele.hasAttribute(DESTROY_METHOD_ATTRIBUTE)) {String destroyMethodName = ele.getAttribute(DESTROY_METHOD_ATTRIBUTE);bd.setDestroyMethodName(destroyMethodName);}else {//没有destroy-method属性,设置默认DestroyMethodif (this.defaults.getDestroyMethod() != null) {bd.setDestroyMethodName(this.defaults.getDestroyMethod());bd.setEnforceDestroyMethod(false);}}//设置factory-method属性if (ele.hasAttribute(FACTORY_METHOD_ATTRIBUTE)) {bd.setFactoryMethodName(ele.getAttribute(FACTORY_METHOD_ATTRIBUTE));}//设置factory-bean属性if (ele.hasAttribute(FACTORY_BEAN_ATTRIBUTE)) {bd.setFactoryBeanName(ele.getAttribute(FACTORY_BEAN_ATTRIBUTE));}return bd;
}

BeanDefinitionParserDelegate.parsePropertyElements(Element beanEle, BeanDefinition bd)

解析bean的属性值property。

public void parsePropertyElements(Element beanEle, BeanDefinition bd) {NodeList nl = beanEle.getChildNodes();for (int i = 0; i < nl.getLength(); i++) {Node node = nl.item(i);if (isCandidateElement(node) && nodeNameEquals(node, PROPERTY_ELEMENT)) {//元素为PROPERTY_ELEMENT才处理parsePropertyElement((Element) node, bd);}}
}public void parsePropertyElement(Element ele, BeanDefinition bd) {//property标签的name属性String propertyName = ele.getAttribute(NAME_ATTRIBUTE);if (!StringUtils.hasLength(propertyName)) {error("Tag 'property' must have a 'name' attribute", ele);return;}//解析property元素进栈this.parseState.push(new PropertyEntry(propertyName));try {//propertyName不能重复nameif (bd.getPropertyValues().contains(propertyName)) {error("Multiple 'property' definitions for property '" + propertyName + "'", ele);return;}//具体解析property元素属性Object val = parsePropertyValue(ele, bd, propertyName);//生成PropertyValue(propertyName:value)PropertyValue pv = new PropertyValue(propertyName, val);//ele有meta子元素,获取meta的key和value属性,设置到PropertyValue中parseMetaElements(ele, pv);pv.setSource(extractSource(ele));//将PropertyValue添加到BeanDefinition中bd.getPropertyValues().addPropertyValue(pv);}finally {      //解析property元素出栈this.parseState.pop();}
}public Object parsePropertyValue(Element ele, BeanDefinition bd, String propertyName) {//如果propertyName为null,则是constructor-arg元素String elementName = (propertyName != null) ?"<property> element for property '" + propertyName + "'" :"<constructor-arg> element";//Property下面都应该只有一个子标签: ref, value, list等.NodeList nl = ele.getChildNodes();Element subElement = null;for (int i = 0; i < nl.getLength(); i++) {Node node = nl.item(i);if (node instanceof Element && !nodeNameEquals(node, DESCRIPTION_ELEMENT) &&!nodeNameEquals(node, META_ELEMENT)) {//除开description和meta标签,子标签最多只能有一个if (subElement != null) {error(elementName + " must not contain more than one sub-element", ele);}else {subElement = (Element) node;}}}//元素属性类型ref、valueboolean hasRefAttribute = ele.hasAttribute(REF_ATTRIBUTE);boolean hasValueAttribute = ele.hasAttribute(VALUE_ATTRIBUTE);//value和ref属性不能同时存在;如果有子元素,则value和ref都不能存在,否则报错if ((hasRefAttribute && hasValueAttribute) ||((hasRefAttribute || hasValueAttribute) && subElement != null)) {error(elementName +" is only allowed to contain either 'ref' attribute OR 'value' attribute OR sub-element", ele);}//元素属性是ref的情况if (hasRefAttribute) {String refName = ele.getAttribute(REF_ATTRIBUTE);if (!StringUtils.hasText(refName)) {error(elementName + " contains empty 'ref' attribute", ele);}//RuntimeBeanReference是ref的不可变占位符类,实际在运行时解析为真正的bean引用实例RuntimeBeanReference ref = new RuntimeBeanReference(refName);ref.setSource(extractSource(ele));return ref;}//元素属性是value的情况else if (hasValueAttribute) {//valueHolder存储String值和目标类型。实际的转换将由BeanFactory执行TypedStringValue valueHolder = new TypedStringValue(ele.getAttribute(VALUE_ATTRIBUTE));valueHolder.setSource(extractSource(ele));return valueHolder;}else if (subElement != null) {//子元素处理return parsePropertySubElement(subElement, bd);}else {//没指定ref或者value或者子元素,返回nullerror(elementName + " must specify a ref or value", ele);return null;}
}//过渡方法
public Object parsePropertySubElement(Element ele, BeanDefinition bd) {return parsePropertySubElement(ele, bd, null);
}//解析property或者constructor-arg标签的子标签,可能为value, ref或者集合
public Object parsePropertySubElement(Element ele, BeanDefinition bd, String defaultValueType) {//如果ele不属于默认命名空间if (!isDefaultNamespace(ele)) {return parseNestedCustomElement(ele, bd);}else if (nodeNameEquals(ele, BEAN_ELEMENT)) {//如果ele是bean类型,嵌套bean处理BeanDefinitionHolder nestedBd = parseBeanDefinitionElement(ele, bd);if (nestedBd != null) {nestedBd = decorateBeanDefinitionIfRequired(ele, nestedBd, bd);}return nestedBd;}else if (nodeNameEquals(ele, REF_ELEMENT)) {//如果ele是ref元素类型String refName = ele.getAttribute(BEAN_REF_ATTRIBUTE);boolean toParent = false;if (!StringUtils.hasLength(refName)) {// A reference to the id of another bean in a parent context.//是对父容器中另一个bean id的引用refName = ele.getAttribute(PARENT_REF_ATTRIBUTE);toParent = true;if (!StringUtils.hasLength(refName)) {error("'bean' or 'parent' is required for <ref> element", ele);return null;}}if (!StringUtils.hasText(refName)) {error("<ref> element contains empty target attribute", ele);return null;}//生成引用占位类RuntimeBeanReference ref = new RuntimeBeanReference(refName, toParent);ref.setSource(extractSource(ele));return ref;}else if (nodeNameEquals(ele, IDREF_ELEMENT)) {//如果是idref引用return parseIdRefElement(ele);}else if (nodeNameEquals(ele, VALUE_ELEMENT)) {//如果是value元素return parseValueElement(ele, defaultValueType);}else if (nodeNameEquals(ele, NULL_ELEMENT)) {//如果是null元素,用null做值处理TypedStringValue nullHolder = new TypedStringValue(null);nullHolder.setSource(extractSource(ele));return nullHolder;}else if (nodeNameEquals(ele, ARRAY_ELEMENT)) {//如果是array元素return parseArrayElement(ele, bd);}    else if (nodeNameEquals(ele, LIST_ELEMENT)) {//如果是list元素return parseListElement(ele, bd);}else if (nodeNameEquals(ele, SET_ELEMENT)) {//如果是set元素return parseSetElement(ele, bd);}else if (nodeNameEquals(ele, MAP_ELEMENT)) {//如果是map元素return parseMapElement(ele, bd);}    else if (nodeNameEquals(ele, PROPS_ELEMENT)) {//如果是props元素return parsePropsElement(ele);}else {//以上都不是,报错error("Unknown property sub-element: [" + ele.getNodeName() + "]", ele);return null;}
}

每种类型元素的解析简单,不再继续跟踪。

BeanDefinitionParserDelegate.decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder originalDef)

按需装饰BeanDefinition。

//过渡方法
public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder) {return decorateBeanDefinitionIfRequired(ele, definitionHolder, null);
}//真正装饰BeanDefinition实现
public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder, BeanDefinition containingBd) {BeanDefinitionHolder finalDefinition = definitionHolder;// Decorate based on custom attributes first.// 基于属性进行NamedNodeMap attributes = ele.getAttributes();for (int i = 0; i < attributes.getLength(); i++) {Node node = attributes.item(i);finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);}// Decorate based on custom nested elements.// 基于嵌套元素进行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;
}
//装饰处理
public BeanDefinitionHolder decorateIfRequired(Node node, BeanDefinitionHolder originalDef, @Nullable BeanDefinition containingBd) {String namespaceUri = getNamespaceURI(node);/*显然,默认命名空间的是不需要装饰的;值装饰非默认命名空间的bean*/if (namespaceUri != null && !isDefaultNamespace(namespaceUri)) {//获取非默认命名空间的处理器NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);if (handler != null) {//装饰处理BeanDefinitionHolder decorated =handler.decorate(node, originalDef, new ParserContext(this.readerContext, this, containingBd));if (decorated != null) {return decorated;}}else if (namespaceUri.startsWith("http://www.springframework.org/schema/")) {error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", node);}else {// A custom namespace, not to be handled by Spring - maybe "xml:...".if (logger.isDebugEnabled()) {logger.debug("No Spring NamespaceHandler found for XML schema namespace [" + namespaceUri + "]");}}}return originalDef;}

BeanDefinitionReaderUtils.registerBeanDefinition(BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)

注册BeanDefinition到容器的BeanFactory

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

说明:registry实际是BeanFactory,registry.registerBeanDefinition和registry.registerAlias实现在DefaultListableBeanFactory,比较简单就不再跟踪。

到此BeanDefinition如何解析跟踪完成,如何使用参见:一起学SF框架系列5.7-模块Beans-BeanDefinition使用

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/15739.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring系列2 -- Spring的创建和使用

Spring 就是⼀个包含了众多工具方法的 IOC容器。既然是容器那么它就具备两个最基本的功能&#xff1a; 将对象存储到容器&#xff08;Spring&#xff09;中&#xff1b;从容器中将对象取出来。 在Java中对象也叫做Bean&#xff0c;后续我们就把对象称之为Bean&#xff1b; 目录…

4.5 x64dbg 探索钩子劫持技术

钩子劫持技术是计算机编程中的一种技术&#xff0c;它们可以让开发者拦截系统函数或应用程序函数的调用&#xff0c;并在函数调用前或调用后执行自定义代码&#xff0c;钩子劫持技术通常用于病毒和恶意软件&#xff0c;也可以让开发者扩展或修改系统函数的功能&#xff0c;从而…

ARM64学习笔记---建立异常向量表(二)

源码: #include "mm.h" #include "sysregs.h".section .rodata .align 3 .globl el_string1 el_string1:.string "Booting at EL".section ".text.boot" .globl _start _start://读取mpidr_el1寄存器的值&#xff0c;该寄存器决定了…

高级运维开发工程师带你处理linux木马(挖矿病毒)实战例子

一、事态描述 centos7测试服务器&#xff0c;突然密码登不上去了&#xff0c;然后CPU占100%。已经猜到&#xff0c;登录密码过于简单&#xff0c;密码被破解挂马了。大概率是CPU挖矿病毒。 二、重置centos7登录root密码 步骤1 启动Linux Centos7系统&#xff0c;当出现如下画…

DevOps系列文章 之 Java使用jgit管理git仓库

最近设计基于gitops新的CICD方案,需要通过java读写git仓库&#xff0c;这里简单记录下。 在jgit中&#xff0c;存在最核心的三个组件&#xff1a;Git类&#xff0c;Repository类。Git类中包含了push commit之类的常见git操作&#xff0c;而Repository则实现了仓库的初始化和基…

使用黑盒测试在 Go 中重写 Bash 脚本

目录 前言&#xff1a; 开始 准备工作 描述行为&#xff1a;Bats 简介 行为描述&#xff1a;陷阱 描述行为&#xff1a;设计测试 重写&#xff1a;让我们开始用go吧&#xff01; 重构和更新&#xff1a;实现胜利 结论 前言&#xff1a; 使用黑盒测试在Go中重写Bash脚本…

百度--搜索引擎是怎么实现的--如何制作一个搜索浏览器

1.搜索引擎是怎么实现的&#xff1f; 搜索引擎是通过以下步骤实现的&#xff1a; 网页抓取&#xff08;Crawling&#xff09;&#xff1a;搜索引擎会使用网络爬虫&#xff08;Web Crawler&#xff09;自动地从互联网上抓取网页内容。爬虫按照一定的规则遍历网页并提取网页内容…

做题遇见的PHP函数汇总

mb_substr函数 mb_substr() 函数返回字符串的一部分&#xff0c;之前学过 substr() 函数&#xff0c;它只针对英文字符&#xff0c;如果要分割的中文文字则需要使用 mb_substr() 语法&#xff1a; mb_substr ( $str ,$start [, $length NULL [, $encoding mb_encoding() ]] …

docker中运行RabbitMq的启用插件指南

我们使用 Docker 来运行 RabbitMQ&#xff0c;有时需要启用一些插件&#xff0c;这个与正常安装的启用插件的步骤会有所不同。以下是在 Docker 中启用 RabbitMQ 插件的一般步骤&#xff1a; 首先&#xff0c;确认已经将 rabbitmq_delayed_message_exchange-3.12.0.ez 插件文件复…

adb: failed to install .\xxxxxx.apk: Failure [INSTALL_FAILED_USER_RESTRICTED

开发者模式和USB调试均已打开&#xff0c;adb安装时报错。看了一下&#xff0c;小米手机还需要开启USB安装才行。 问题已解决

《永劫无间》免费!《永劫无间》配置要求如何?

《永劫无间》于2021年7月8日开启不删档测试&#xff0c;8月12日在Steam全球公测&#xff0c;终于赶在两周年&#xff0c;2023年7月14日的炎炎夏日转为免费游戏。在《永劫无间》中&#xff0c;玩家可以选择不同的英雄&#xff0c;每个英雄都有独特的技能和风格&#xff0c;通过搜…

深入解析Redis的LRU与LFU算法实现

作者&#xff1a;vivo 互联网服务器团队 - Luo Jianxin 重点介绍了Redis的LRU与LFU算法实现&#xff0c;并分析总结了两种算法的实现效果以及存在的问题。 一、前言 Redis是一款基于内存的高性能NoSQL数据库&#xff0c;数据都缓存在内存里&#xff0c; 这使得Redis可以每秒轻…