资讯专栏INFORMATION COLUMN

Spring源码阅读——ClassPathXmlApplicationContext(二)

Nekron / 724人阅读

摘要:在上一篇文章中,分析了容器的创建,加载资源文件,将资源文件读取为。将文件中的注册定义的对象。在中对属性的解析委托给这个代理类来实现的。首先,获取节点。

在上一篇文章中,分析了ApplicationContext容器的创建,加载资源文件,将资源文件读取为Document。spring将xml文件中的Bean注册spring定义的BeanDefinition对象。在DefaultBeanDefinitionDocumentReader中对Document属性的解析委托给BeanDefinitionParserDelegate这个代理类来实现的。

Bean注册前的准备

DefaultBeanDefinitionDocumentReader的registerBeanDefinitions方法实现如下,首先获取Document的根元素,接着调用doRegisterBeanDefinitions(root)进行注册

</>复制代码

  1. public void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {
  2. this.readerContext = readerContext;
  3. //获取根元素
  4. Element root = doc.getDocumentElement();
  5. //注册BeanDefinition
  6. doRegisterBeanDefinitions(root);
  7. }

doRegisterBeanDefinitions(root)方法的实现如下:

</>复制代码

  1. protected void doRegisterBeanDefinitions(Element root) {
  2. //获取代理类
  3. BeanDefinitionParserDelegate parent = this.delegate;
  4. this.delegate = createDelegate(getReaderContext(), root, parent);
  5. //是否为默认命名空间
  6. if (this.delegate.isDefaultNamespace(root)) {
  7. String profileSpec = root.getAttribute(PROFILE_ATTRIBUTE);
  8. //是否有profile属性
  9. if (StringUtils.hasText(profileSpec)) {
  10. String[] specifiedProfiles = StringUtils.tokenizeToStringArray(
  11. profileSpec, BeanDefinitionParserDelegate.MULTI_VALUE_ATTRIBUTE_DELIMITERS);
  12. if (!getReaderContext().getEnvironment().acceptsProfiles(specifiedProfiles)) {
  13. return;
  14. }
  15. }
  16. }
  17. preProcessXml(root);
  18. //解析BeanDefinition
  19. parseBeanDefinitions(root, this.delegate);
  20. postProcessXml(root);
  21. this.delegate = parent;
  22. }
根据不同节点名进行解析

parseBeanDefinitions(root, this.delegate)方法是解析根源素下定义的每一个bean。首先,获取节点List。其次,判断每个元素是否为默认的命名空间中的元素,然后交给不同的方法去解析,具体的实现如下:

</>复制代码

  1. protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {
  2. //如果根元素为默认命名空间中的元素
  3. if (delegate.isDefaultNamespace(root)) {
  4. //获取字元素List
  5. NodeList nl = root.getChildNodes();
  6. for (int i = 0; i < nl.getLength(); i++) {
  7. Node node = nl.item(i);
  8. if (node instanceof Element) {
  9. Element ele = (Element) node;
  10. //判断此元素是否为默认命名空间的元素
  11. if (delegate.isDefaultNamespace(ele)) {
  12. parseDefaultElement(ele, delegate);
  13. }
  14. else {
  15. delegate.parseCustomElement(ele);
  16. }
  17. }
  18. }
  19. }
  20. else {
  21. delegate.parseCustomElement(root);
  22. }
  23. }
spring默认命名空间节点的解析

下面,我们首先看spring默认命名空间元素的解析过程,parseDefaultElement(ele, delegate)方法的实现如下:

</>复制代码

  1. private void parseDefaultElement(Element ele, BeanDefinitionParserDelegate delegate) {
  2. //节点名为import
  3. if (delegate.nodeNameEquals(ele, IMPORT_ELEMENT)) {
  4. importBeanDefinitionResource(ele);
  5. }
  6. //节点名为alias
  7. else if (delegate.nodeNameEquals(ele, ALIAS_ELEMENT)) {
  8. processAliasRegistration(ele);
  9. }
  10. //节点名为bean
  11. else if (delegate.nodeNameEquals(ele, BEAN_ELEMENT)) {
  12. processBeanDefinition(ele, delegate);
  13. }
  14. //节点名为beans
  15. else if (delegate.nodeNameEquals(ele, NESTED_BEANS_ELEMENT)) {
  16. // recurse 循环调用
  17. doRegisterBeanDefinitions(ele);
  18. }
  19. }

import节点的解析

import元素是引入其他的配置文件,resource属性是配置文件的路径,importBeanDefinitionResource(ele)方法的实现如下,省略了异常处理代码:

</>复制代码

  1. protected void importBeanDefinitionResource(Element ele) {
  2. //获取resource属性值,即其他配置文件的路径
  3. String location = ele.getAttribute(RESOURCE_ATTRIBUTE);
  4. // 解析路径,如"${user.dir}" 这样的路径是从在propertie文件中加载的
  5. location = getReaderContext().getEnvironment().resolveRequiredPlaceholders(location);
  6. Set actualResources = new LinkedHashSet<>(4);
  7. // 判断location 是绝对路径还是相对路径
  8. boolean absoluteLocation = false;
  9. absoluteLocation = ResourcePatternUtils.isUrl(location) || ResourceUtils.toURI(location).isAbsolute();
  10. // 绝对路径
  11. if (absoluteLocation) {
  12. //调用loadBeanDefinitions(location, actualResources)方法解析此配置文件
  13. int importCount = getReaderContext().getReader().loadBeanDefinitions(location, actualResources);
  14. }
  15. }
  16. //相对路径
  17. else {
  18. int importCount;
  19. Resource relativeResource = getReaderContext().getResource().createRelative(location);
  20. if (relativeResource.exists()) {
  21. //调用loadBeanDefinitions(relativeResource)方法解析此配置文件
  22. importCount = getReaderContext().getReader().loadBeanDefinitions(relativeResource);
  23. actualResources.add(relativeResource);
  24. }
  25. else {
  26. String baseLocation = getReaderContext().getResource().getURL().toString();
  27. importCount = getReaderContext().getReader().loadBeanDefinitions(
  28. StringUtils.applyRelativePath(baseLocation, location), actualResources);
  29. }
  30. }
  31. }
  32. //广播Import元素处理事件
  33. Resource[] actResArray = actualResources.toArray(new Resource[0]);
  34. getReaderContext().fireImportProcessed(location, actResArray, extractSource(ele));
  35. }

alias节点的解析

下面介绍alias元素的方法,processAliasRegistration(ele)方法的实现如下:

</>复制代码

  1. protected void processAliasRegistration(Element ele) {
  2. //获取name属性
  3. String name = ele.getAttribute(NAME_ATTRIBUTE);
  4. //获取alias属性
  5. String alias = ele.getAttribute(ALIAS_ATTRIBUTE);
  6. boolean valid = true;
  7. if (!StringUtils.hasText(name)) {
  8. getReaderContext().error("Name must not be empty", ele);
  9. valid = false;
  10. }
  11. if (!StringUtils.hasText(alias)) {
  12. getReaderContext().error("Alias must not be empty", ele);
  13. valid = false;
  14. }
  15. if (valid) {
  16. //调用SimpleAliasRegistry类的registerAlias(name, alias)进行注册
  17. getReaderContext().getRegistry().registerAlias(name, alias);
  18. getReaderContext().fireAliasRegistered(name, alias, extractSource(ele));
  19. }
  20. }

在SimpleAliasRegistry中定义了aliasMap来存储alias和name的关系,具体实现如下:

</>复制代码

  1. private final Map aliasMap = new ConcurrentHashMap<>(16);
  2. public void registerAlias(String name, String alias) {
  3. Assert.hasText(name, ""name" must not be empty");
  4. Assert.hasText(alias, ""alias" must not be empty");
  5. synchronized (this.aliasMap) {
  6. //如果alias和name相等,将此关系移除
  7. if (alias.equals(name)) {
  8. this.aliasMap.remove(alias);
  9. }
  10. else {
  11. //先从aliasMap获取key为alias的beanName
  12. String registeredName = this.aliasMap.get(alias);
  13. if (registeredName != null) {
  14. //如果已存在,return
  15. if (registeredName.equals(name)) {
  16. // An existing alias - no need to re-register
  17. return;
  18. }
  19. //如果不存在,判断alias是否可以继承,默认是true
  20. if (!allowAliasOverriding()) {
  21. throw new IllegalStateException("Cannot register alias "" + alias + "" for name "" +
  22. name + "": It is already registered for name "" + registeredName + "".");
  23. }
  24. }
  25. //检查是否存在循环依赖
  26. checkForAliasCircle(name, alias);
  27. //注册alias和name
  28. this.aliasMap.put(alias, name);
  29. }
  30. }
  31. }

bean节点的解析

processBeanDefinition(ele, delegate)实现如下:

</>复制代码

  1. protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {
  2. //解析bean元素,创建BeanDefinitionHolder实例
  3. BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
  4. if (bdHolder != null) {
  5. //完成必须的装配
  6. bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);
  7. try {
  8. // 进行最终的注册bean
  9. BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());
  10. }
  11. catch (BeanDefinitionStoreException ex) {
  12. getReaderContext().error("Failed to register bean definition with name "" +
  13. bdHolder.getBeanName() + """, ele, ex);
  14. }
  15. // Send registration event.
  16. getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));
  17. }
  18. }

bean元素的解析和注册相对复杂,在下一节中讨论。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/71125.html

相关文章

  • Spring源码阅读——ClassPathXmlApplicationContext

    摘要:在上一篇文章中,分析了容器的创建,加载资源文件,将资源文件读取为。将文件中的注册定义的对象。在中对属性的解析委托给这个代理类来实现的。首先,获取节点。 在上一篇文章中,分析了ApplicationContext容器的创建,加载资源文件,将资源文件读取为Document。spring将xml文件中的Bean注册spring定义的BeanDefinition对象。在DefaultBeanD...

    linkFly 评论0 收藏0
  • Spring源码阅读——ClassPathXmlApplicationContext

    摘要:在上一篇文章中,分析了容器的创建,加载资源文件,将资源文件读取为。将文件中的注册定义的对象。在中对属性的解析委托给这个代理类来实现的。首先,获取节点。 在上一篇文章中,分析了ApplicationContext容器的创建,加载资源文件,将资源文件读取为Document。spring将xml文件中的Bean注册spring定义的BeanDefinition对象。在DefaultBeanD...

    MorePainMoreGain 评论0 收藏0
  • Spring源码阅读——ClassPathXmlApplicationContext(四)

    摘要:在的方法中,遍历每一个节点,判断是否为默认命名空间中的节点,如果是非默认命名空间的,调用方法进行处理。在学习自定义标签解析之前,先写一个自定义标签的。 在DefaultBeanDefinitionDocumentReader的parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate)方法中,遍历每一...

    silenceboy 评论0 收藏0
  • Spring源码阅读——ClassPathXmlApplicationContext(四)

    摘要:在的方法中,遍历每一个节点,判断是否为默认命名空间中的节点,如果是非默认命名空间的,调用方法进行处理。在学习自定义标签解析之前,先写一个自定义标签的。 在DefaultBeanDefinitionDocumentReader的parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate)方法中,遍历每一...

    wmui 评论0 收藏0
  • Spring源码阅读——ClassPathXmlApplicationContext(四)

    摘要:在的方法中,遍历每一个节点,判断是否为默认命名空间中的节点,如果是非默认命名空间的,调用方法进行处理。在学习自定义标签解析之前,先写一个自定义标签的。 在DefaultBeanDefinitionDocumentReader的parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate)方法中,遍历每一...

    ixlei 评论0 收藏0

发表评论

0条评论

Nekron

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<