摘要:在上一篇源碼閱讀二文章的最后,需要解析元素,創(chuàng)建實例完成必須的裝配和進行最終的注冊來完成元素的解析和注冊,下面分別閱讀三步的源碼。
在上一篇Spring源碼閱讀——ClassPathXmlApplicationContext(二)文章的最后,需要解析bean元素,創(chuàng)建BeanDefinitionHolder實例、完成必須的裝配和進行最終的注冊bean來完成bean元素的解析和注冊,下面分別閱讀三步的源碼。
創(chuàng)建BeanDefinitionHolder實例BeanDefinitionHolder的創(chuàng)建是委托給BeanDefinitionParserDelegate這個代理類的parseBeanDefinitionElement(Element ele, @Nullable BeanDefinition containingBean)方法來完成的,此方法的實現(xiàn)如下:
public BeanDefinitionHolder parseBeanDefinitionElement(Element ele, @Nullable BeanDefinition containingBean) { // 獲取id屬性值 String id = ele.getAttribute(ID_ATTRIBUTE); // 獲取name屬性值 String nameAttr = ele.getAttribute(NAME_ATTRIBUTE); //解析name屬性值,將所有name放入List中 Listaliases = new ArrayList<>(); if (StringUtils.hasLength(nameAttr)) { String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr, MULTI_VALUE_ATTRIBUTE_DELIMITERS); aliases.addAll(Arrays.asList(nameArr)); } // 賦值beanName為id String beanName = id; if (!StringUtils.hasText(beanName) && !aliases.isEmpty()) { // 如果id為空,且name屬性不為空,取第一個name為beanName 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) { // 檢查唯一性 checkNameUniqueness(beanName, aliases, ele); } // 解析bean元素 AbstractBeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName, containingBean); if (beanDefinition != null) { // 如果beanName為空 if (!StringUtils.hasText(beanName)) { try { if (containingBean != null) { // 生成默認的beanName beanName = BeanDefinitionReaderUtils.generateBeanName( beanDefinition, this.readerContext.getRegistry(), true); } else { // 生成默認的beanName beanName = this.readerContext.generateBeanName(beanDefinition); // 通過beanClass生成一個alias String beanClassName = beanDefinition.getBeanClassName(); if (beanClassName != null && beanName.startsWith(beanClassName) && beanName.length() > beanClassName.length() && !this.readerContext.getRegistry().isBeanNameInUse(beanClassName)) { aliases.add(beanClassName); } } } catch (Exception ex) { error(ex.getMessage(), ele); return null; } } // 得到別名數(shù)組 String[] aliasesArray = StringUtils.toStringArray(aliases); // 創(chuàng)建BeanDefinitionHolder對象并返回 return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray); } return null; }
在上述源碼中,調(diào)用了parseBeanDefinitionElement(ele, beanName, containingBean)方法創(chuàng)建AbstractBeanDefinition 實例,下面是此方法的實現(xiàn):
public AbstractBeanDefinition parseBeanDefinitionElement( Element ele, String beanName, @Nullable BeanDefinition containingBean) { // 根據(jù)beanName創(chuàng)建BeanEntry實體,并且push BeanEntry this.parseState.push(new BeanEntry(beanName)); // 獲取class屬性值 String className = null; if (ele.hasAttribute(CLASS_ATTRIBUTE)) { className = ele.getAttribute(CLASS_ATTRIBUTE).trim(); } // 獲取parent屬性值 String parent = null; if (ele.hasAttribute(PARENT_ATTRIBUTE)) { parent = ele.getAttribute(PARENT_ATTRIBUTE); } try { // 創(chuàng)建AbstractBeanDefinition實例 AbstractBeanDefinition bd = createBeanDefinition(className, parent); // 設(shè)置AbstractBeanDefinition實例的其他各種屬性 parseBeanDefinitionAttributes(ele, beanName, containingBean, bd); // 獲取子元素description bd.setDescription(DomUtils.getChildElementValueByTagName(ele, DESCRIPTION_ELEMENT)); // 獲取子元素meta parseMetaElements(ele, bd); // 解析子元素lookup-method parseLookupOverrideSubElements(ele, bd.getMethodOverrides()); // 解析子元素replaced-method parseReplacedMethodSubElements(ele, bd.getMethodOverrides()); // 解析子元素constructor-arg parseConstructorArgElements(ele, bd); // 解析子元素property parsePropertyElements(ele, bd); // 解析子元素qualifier parseQualifierElements(ele, bd); 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 { // pop BeanEntry this.parseState.pop(); } return null; }如果有非默認命名空間,繼續(xù)裝飾此實例
decorateBeanDefinitionIfRequired(ele, bdHolder)接口的最終實現(xiàn)如下:
public BeanDefinitionHolder decorateBeanDefinitionIfRequired( Element ele, BeanDefinitionHolder definitionHolder, @Nullable BeanDefinition containingBd) { BeanDefinitionHolder finalDefinition = definitionHolder; // 根據(jù)自定義屬性裝飾 // 獲取所有屬性 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; }
此方法在存在非默認命名空間時,解析自定義的屬性和嵌套元素,decorateIfRequired(node, finalDefinition, containingBd)方法的實現(xiàn)如下:
public BeanDefinitionHolder decorateIfRequired( Node node, BeanDefinitionHolder originalDef, @Nullable BeanDefinition containingBd) { // 獲取命名空間uri String namespaceUri = getNamespaceURI(node); // 如果不是默認命名空間uri if (namespaceUri != null && !isDefaultNamespace(namespaceUri)) { // 獲取命名空間handler NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri); if (handler != null) { // 調(diào)用此handler 的decorate方法創(chuàng)建BeanDefinitionHolder實例 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/")) { 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; }注冊最終的實例
接下來是往注冊表中注冊bean,BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry())的實現(xiàn)如下:
public static void registerBeanDefinition( BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry) throws BeanDefinitionStoreException { String beanName = definitionHolder.getBeanName(); // beanName 作為key,注冊bean definition registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition()); // beanName 作為key,注冊alias String[] aliases = definitionHolder.getAliases(); if (aliases != null) { for (String alias : aliases) { registry.registerAlias(beanName, alias); } } }
在SimpleBeanDefinitionRegistry類中,定義了注冊bean definition的Mam對象,map初始大小為64,如下:
private final Map
registerBeanDefinition(beanName, definitionHolder.getBeanDefinition())
public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition) throws BeanDefinitionStoreException { Assert.hasText(beanName, ""beanName" must not be empty"); Assert.notNull(beanDefinition, "BeanDefinition must not be null"); this.beanDefinitionMap.put(beanName, beanDefinition); }
通過這篇文章,學(xué)習(xí)了ClassPathXmlApplicationContext的啟動解析默認命名空間中元素的解析過程,下面繼續(xù)學(xué)習(xí)自定義命名空間中bean的解析。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/71126.html
摘要:在上一篇源碼閱讀二文章的最后,需要解析元素,創(chuàng)建實例完成必須的裝配和進行最終的注冊來完成元素的解析和注冊,下面分別閱讀三步的源碼。 在上一篇Spring源碼閱讀——ClassPathXmlApplicationContext(二)文章的最后,需要解析bean元素,創(chuàng)建BeanDefinitionHolder實例、完成必須的裝配和進行最終的注冊bean來完成bean元素的解析和注冊,下面...
摘要:在上一篇源碼閱讀二文章的最后,需要解析元素,創(chuàng)建實例完成必須的裝配和進行最終的注冊來完成元素的解析和注冊,下面分別閱讀三步的源碼。 在上一篇Spring源碼閱讀——ClassPathXmlApplicationContext(二)文章的最后,需要解析bean元素,創(chuàng)建BeanDefinitionHolder實例、完成必須的裝配和進行最終的注冊bean來完成bean元素的解析和注冊,下面...
摘要:的繼承關(guān)系繼承了,實現(xiàn)了接口。是所有容器的頂級接口,中所有容器都是基于的。方法創(chuàng)建一個新的容器。在本方法中,最重要的是,調(diào)用這個方法解析配置文件,注冊。 ClassPathXmlApplicationContext的繼承關(guān)系 ClassPathXmlApplicationContext繼承了AbstractXmlApplicationContext,實現(xiàn)了ApplicationCont...
摘要:在的方法中,遍歷每一個節(jié)點,判斷是否為默認命名空間中的節(jié)點,如果是非默認命名空間的,調(diào)用方法進行處理。在學(xué)習(xí)自定義標簽解析之前,先寫一個自定義標簽的。 在DefaultBeanDefinitionDocumentReader的parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate)方法中,遍歷每一...
摘要:在的方法中,遍歷每一個節(jié)點,判斷是否為默認命名空間中的節(jié)點,如果是非默認命名空間的,調(diào)用方法進行處理。在學(xué)習(xí)自定義標簽解析之前,先寫一個自定義標簽的。 在DefaultBeanDefinitionDocumentReader的parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate)方法中,遍歷每一...
閱讀 1194·2021-10-11 10:59
閱讀 1974·2021-09-29 09:44
閱讀 863·2021-09-01 10:32
閱讀 1436·2019-08-30 14:21
閱讀 1880·2019-08-29 15:39
閱讀 2985·2019-08-29 13:45
閱讀 3541·2019-08-29 13:27
閱讀 2015·2019-08-29 12:27