13, [] Sike 봄 - 해상도 태그의 IOC의이 : 구문 분석 사용자 지정 라벨

앞의 네 기사를 포함하여 콩 기본 해상도 프로세스 태그, 분석  기본 속성 , 여섯 하위 요소를 ( meta,, lookup-method,, replaced-method, ),constructor-arg 우리가 잊어 버린 곳에서의 4 개 개의 기사에 더 많은 콘텐츠, 분할, 관련 의. 자,가 보자 모양 다시 에 있습니다.propertyqualifier

DefaultBeanDefinitionDocumentReader  #processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) 방법에 대한 책임이  <bean> 레이블의 분석 :

  • 확인 프로세스에서 첫 번째 전화  BeanDefinitionParserDelegate#parseBeanDefinitionElement(Element ele) 완전한하는 방법 기본 레이블 해결합니다.
  • 성공적으로 해결 (귀국일 경우  bdHolder != null ), 다음 호출  BeanDefinitionParserDelegate#decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder) 전체하는 방법을 사용자 정의 태그의 구문 분석 요소를.

1. decorateBeanDefinitionIfRequired

지난 4 개 기사, 기본 레이블 구문 분석을 분석 한 이 문서는 사용자 지정 레이블 구문 분석에서 분석 있도록 . 다음과 같이 코드입니다 :

// BeanDefinitionParserDelegate.java

public BeanDefinitionHolder decorateBeanDefinitionIfRequired(
        Element ele, BeanDefinitionHolder definitionHolder, @Nullable BeanDefinition containingBd) {

    BeanDefinitionHolder finalDefinition = definitionHolder;

    // <1> 遍历属性,查看是否有适用于装饰的【属性】
    // 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);
    }

    // <2> 遍历子节点,查看是否有适用于修饰的【子节点】
    // 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;
}
  • <1> 그리고  <2> 에서이 통과되어, 전자는 (속성을 통과한다  attributes 통과 자식 노드 (인)  childNodes ), 마지막 호출  #decorateIfRequired(Node node, BeanDefinitionHolder originalDef, BeanDefinition containingBd) 방법, 노드에 대응 장식 (노드). 자세한 분석을 참조하십시오  "2 decorateIfRequired을」  .

2. decorateIfRequired

#decorateIfRequired(Node node, BeanDefinitionHolder originalDef, BeanDefinition containingBd) 방법, 대응 노드 장식품 (노드). 다음과 같이 코드입니다 :

// BeanDefinitionParserDelegate.java

public BeanDefinitionHolder decorateIfRequired(
        Node node, BeanDefinitionHolder originalDef, @Nullable BeanDefinition containingBd) {
    // <1> 获取自定义标签的命名空间
    String namespaceUri = getNamespaceURI(node);
    // <2> 过滤掉默认命名标签
    if (namespaceUri != null && !isDefaultNamespace(namespaceUri)) {
        // <2> 获取相应的处理器
        NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
        if (handler != null) {
            // <3> 进行装饰处理
            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;
}
  • 在 <1> 处,首先获取自定义标签的命名空间。
  • 在 <2> 处,如果不是默认的命名空间,则根据该命名空间获取相应的处理器。
  • 在 <3> 处,如果处理器存在,则进行装饰处理。

上述过程的详细解析,见 《【死磕 Spring】—— IoC 之解析 标签:解析自定义标签》 一文。

3. 小结

至此,BeanDefinition 的解析过程已经全部完成了,下面做一个简要的总结:

解析 BeanDefinition 的入口在 DefaultBeanDefinitionDocumentReader 的#parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) 方法。该方法会根据命令空间来判断标签是默认标签还是自定义标签,其中:

  • 默认标签,由 #parseDefaultElement(Element ele, BeanDefinitionParserDelegate delegate) 方法来实现
  • 自定义标签,由 BeanDefinitionParserDelegate 的 #parseCustomElement(Element ele, @Nullable BeanDefinition containingBd) 方法来实现。

在默认标签解析中,会根据标签名称的不同进行 importaliasbeanbeans 四大标签进行处理。其中 bean 标签的解析为核心,它由 processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) 方法实现。

processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) 方法,开始进入解析核心工作,分为三步:

  1. 解析默认标签的默认标签:BeanDefinitionParserDelegate#parseBeanDefinitionElement(Element ele, ...) 方法。该方法会依次解析 <bean> 标签的属性、各个子元素,解析完成后返回一个 GenericBeanDefinition 实例对象。
  2. 解析默认标签下的自定义标签:BeanDefinitionParserDelegate#decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder) 方法。
  3. 注册解析的 BeanDefinition:BeanDefinitionReaderUtils#registerBeanDefinition(BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry) 方法。
发布了12 篇原创文章 · 获赞 1 · 访问量 753

추천

출처blog.csdn.net/a1290123825/article/details/88382678