`

spring 源码分析--IOC容器初始化六

阅读更多

 上一节将xml文档解析为DOM ,并且创建了一个 BeanDefinitionParserDelegate 类型的对象,在这一节,将使用这个对象来完成对bean的装载工作。

 

2.1.1.1 parseBeanDefinitions (root, delegate): 该方法体完成注册过程。

 

====================================================================

    /**

      * Process the given bean element, parsing the bean definition

      * and registering it with the registry . 这里对我们最熟悉的 bean 元素进行处理

      */

protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {

/**

* 委托给 BeanDefinitionParserDelegate 来完成对 bean 元素的处理,这个类包含了具

* 体的 bean 解析的过程。把解析 bean 文件得到的信息放到 BeanDefinition 里,他是 bean

* 信息的主要载体,也是 IOC 容器的管理对象。  

*/

BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele) ;

if (bdHolder != null ) {

    bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);

try {

    // 这里是向 IOC 容器注册,实际上是放到 IOC 容器的一个 map

BeanDefinitionReaderUtils.registerBeanDefinition (bdHolder, getReaderContext().getRegistry());

} catch (BeanDefinitionStoreException ex) {

getReaderContext().error( "Failed to register bean definition with name '" +bdHolder.getBeanName() + "'" , ele, ex);

}

      // 这里向 IOC 容器发送事件,表示解析和注册完成。

    getReaderContext().fireComponentRegistered( new BeanComponentDefinition(bdHolder));

    }

}

====================================================================

 

2.1.1.1.1 parseBeanDefinitionElement(Element ele) 定义在类 BeanDefinitionParserDelegate 中:定义如下:

 

====================================================================

public BeanDefinitionHolder parseBeanDefinitionElement (Element ele, BeanDefinition containingBean) {

//ID_ATTRIBUTE = "id"

    String id = ele.getAttribute( ID_ATTRIBUTE );

//NAME_ATTRIBUTE = "name"

    String nameAttr = ele.getAttribute( NAME_ATTRIBUTE );

    List aliases = new ArrayList ();

    if (StringUtils.hasLength (nameAttr)) {

// BEAN_NAME_DELIMITERS = ",; "

String[] nameArr = StringUtils.tokenizeToStringArray (nameAttr, BEAN_NAME_DELIMITERS );

// 添加别名

       aliases.addAll(Arrays.asList (nameArr)) ;

    }

    String beanName = id;

// 如果没有定义 id ,但是定义了 name 属性,则把 name 属性中的第一个做 bean 的名字。

    if (!StringUtils.hasText (beanName) && !aliases.isEmpty()) {

       beanName = (String) aliases.remove(0);

       if ( logger .isDebugEnabled()) {

           logger .debug( "No XML 'id' specified - using '" + beanName +

                     "' as bean name and " + aliases + " as aliases" );

       }

    }

// 默认为 null 

    if (containingBean == null ) {

// 保证唯一性,

       checkNameUniqueness(beanName, aliases, ele) ;

    }

// 自己解析 bean 的定义,出错可能返回 null

AbstractBeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName, containingBean) ;

    if (beanDefinition != null ) {

/**

* 如果没有 beanname 这根据父 bean 或者生产它的工厂名字产生 beanname

*/

       if (!StringUtils.hasText (beanName)) {

           try {

               if (containingBean != null ) {

                  beanName = BeanDefinitionReaderUtils.generateBeanName (

                            beanDefinition, this . readerContext .getRegistry(), true );

              } else {

                  beanName = this . readerContext .generateBeanName(beanDefinition);

    // Register an alias for the plain bean class name, if still possible,

    // if the generator returned the class name plus a suffix.

    // This is expected for Spring 1.2/2.0 backwards compatibility.

                  String beanClassName = beanDefinition.getBeanClassName();

                  if (beanClassName != null &&

                         beanName.startsWith(beanClassName) && beanName.length() > beanClassName.length() &&                         ! this . readerContext .getRegistry().isBeanNameInUse(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);

           return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);

       }

 

       return null ;

    }

====================================================================

 

2.1.1.1.1.1 checkNameUniqueness(beanName, aliases, ele) :保证 beanname 的唯一性,

====================================================================

protected void checkNameUniqueness(String beanName, List aliases, Element beanElement) {

    String foundName = null ;

// 如果 usedNames 集合里面已经有了,报错

if (StringUtils.hasText (beanName) && this . usedNames .contains(beanName)){

           foundName = beanName;

    }

    if (foundName == null ) {

foundName = (String) CollectionUtils.findFirstMatch ( this . usedNames , aliases);

    }

    if (foundName != null ) {

error( "Bean name '" + foundName + "' is already used in this file" , beanElement);

    }

// 如果都 ok 则将当前的 beanname 放到集合中。

    this . usedNames .add(beanName) ;

    this . usedNames .addAll(aliases) ;

}

====================================================================

 

 

2.1.1.1.1.2 parseBeanDefinitionElement(ele, beanName, containingBean)

 

自己解析 bean 的定义。定义如下:

====================================================================

public AbstractBeanDefinition parseBeanDefinitionElement(

           Element ele, String beanName, BeanDefinition containingBean) {

this . parseState .push( new BeanEntry(beanName));

    String className = null ;

    if (ele.hasAttribute( CLASS_ATTRIBUTE )) {

       className = ele.getAttribute( CLASS_ATTRIBUTE ).trim();

    }

    try {

       String parent = null ;

       if (ele.hasAttribute( PARENT_ATTRIBUTE )) {

           parent = ele.getAttribute( PARENT_ATTRIBUTE );

       }

// 通过 BeanDefinitionReaderUtils 类装载 bean 的实例,返回一个

// AbstractBeanDefinition 子类: GenericBeanDefinition

// 设置 beanClass parentName

AbstractBeanDefinition bd = createBeanDefinition(className, parent);

// 设置属性

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

       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 ;

}

 

到这里我们将xml里面关于bean的定义封装到一个AbstractBeanDefinition对象里面,并且进一步封装到 BeanDefinitionHolder类型的对象里。

 

 

本站支持 pay for your wishes

1
1
分享到:
评论

相关推荐

    Spring源码分析_Spring_IOC

    ### Spring源码分析_Spring_IOC:深入理解Spring的IOC容器机制 #### 基本概念与核心作用 在探讨Spring框架的核心组件之一——IOC(Inversion of Control,控制反转)容器之前,首先需要理解它在Spring框架中的角色...

    Spring框架系列(7) - Spring IOC实现原理详解之IOC初始化流程.doc

    Spring 框架系列(7)- Spring IOC 实现原理详解之 IOC 初始化流程 本文将详细解释 Spring 框架中的 IOC...IOC 容器的初始化流程是 Spring 框架中的关键部分,用于将资源配置文件中的信息加载到 IOC 容器中。

    Spring源码解析4章150页+Spring3.2.4中文注释源码

    3、源码分析-IOC容器的初始化 4、源码分析-IOC容器的依赖注入 5、源码分析-IOC容器的高级特性 三阶段 Spring AOP的涉及原理及具体实践 SpringJDBC的涉及原理及二次开发 SpringMVC框架设计原理及手写实现 四阶段 ...

    Spring IOC容器实现分析.pdf 下载

    《Spring IOC容器实现分析》 在Java开发领域,Spring框架无疑是使用最为广泛的轻量级框架之一,其中的核心组件就是IOC(Inversion of Control)容器。本文将深入剖析Spring的IOC容器,理解其工作原理和重要功能,以...

    spring源码解读-地址.txt

    根据提供的文件信息,本次解读将围绕Spring框架的核心概念与源码分析进行展开。Spring框架作为Java企业级开发中不可或缺的一部分,其源码的学习对于深入理解框架机制、提高开发效率具有重要意义。下面,我们将从以下...

    官方原版源码spring-framework-5.1.13.RELEASE.zip

    通过分析BeanFactory,我们可以了解对象的创建、初始化和依赖注入过程。 - **AOP**:面向切面编程允许我们定义横切关注点,如日志记录、事务管理,然后将其模块化为可重用的组件。 - **Data Access/...

    Spring IoC源码深度剖析开源架构源码2021.pdf

    文档可能将深入探讨Spring IoC容器初始化、Bean生命周期管理、依赖注入等关键概念,并以2021年的开源版本为背景进行分析。 从提供的部分文档内容来看,我们可以提炼出以下几个知识点: 1. **BeanFactory与...

    官方原版源码 spring-framework-5.1.15.RELEASE.zip

    Spring Core负责基础的IoC(Inversion of Control)容器,Spring Beans则实现了Bean的生命周期管理和配置,Spring Context则是基于Core之上构建的,提供了一个应用上下文,可以管理Bean并与其他Spring模块集成。...

    官方原版源码spring-framework-5.1.0.RELEASE.zip

    2. **Spring Beans**:实现了IoC容器,通过XML或注解方式配置bean,并负责bean的初始化、装配和销毁。 3. **Spring AOP**:面向切面编程模块,允许定义方法拦截器和切入点,实现代码的解耦和模块化。 4. **Spring ...

    【框架源码篇 01】Spring源码-手写IOC

    5. **Bean管理**:最后,容器会管理Bean的生命周期,包括初始化、使用和销毁。 在实际开发中,Spring使用了更加高效和灵活的方式来实现这些功能,比如使用`ApplicationContext`接口作为IOC容器的入口,以及`...

    官方原版源码 spring-framework-5.2.9.RELEASE.zip

    1. **ApplicationContext**:这是Spring的核心接口,负责初始化、加载配置文件,管理Bean的生命周期。源码中可以看到它是如何解析XML或Java配置,创建并管理Bean的。 2. **BeanFactory**:作为ApplicationContext的...

    spring-framework-2.5-rc2-with-dependencies\spring-framework-2.5-rc2\spring-framework-2.5-rc2源代码

    通过阅读和学习这些源码,开发者可以了解到Spring如何实现IoC容器、AOP代理、事件机制、任务调度等多个关键功能。同时,这也有助于开发者更好地理解和使用Spring提供的API,以及在实际项目中如何定制和扩展Spring。 ...

    官方原版源码spring-framework-5.0.13.RELEASE.zip

    1. **初始化流程**:从`org.springframework.context.support.ClassPathXmlApplicationContext`或`org.springframework.web.context.ContextLoader`开始,理解如何加载配置并创建Bean定义。 2. **依赖注入**:研究`...

    spring-cglib-repack-3.2.6.jar和spring-objenesis-repack-2.6.jar

    在深入探讨Spring框架的核心组件和工作...在分析或运行Spring源码时,确保正确引入这两个库是确保项目正常运行的关键。通过深入了解这些工具库,开发者可以更好地理解和运用Spring框架,从而提高开发效率和应用性能。

    Spring2.5.6源代码分析(一):IOC容器

    IoC容器是Spring框架的心脏,它负责管理对象的生命周期和依赖关系,使得开发者能够实现松耦合和高可测试性的应用程序。 首先,我们来理解什么是IoC。IoC,也被称为依赖注入(Dependency Injection),是一种设计...

    spring-framework-5.3.29.tar.gz

    通过XML配置文件或Java注解,我们可以定义bean的创建、初始化和销毁过程,实现依赖注入,从而降低代码的耦合度。 2. AOP:Spring的AOP模块允许开发者定义横切关注点,如日志记录、事务管理等,将这些关注点与业务...

    Spring 5.2.9的IOC核心jar包

    通过配置文件或者注解,开发者可以定义对象及其依赖关系,Spring容器会自动管理这些对象的生命周期,包括实例化、初始化、装配和销毁。 **Bean** 在Spring中,业务逻辑的组件被称为Bean。Bean是Spring容器管理的...

    coco-spring-ioc

    《简易Spring-ioc详解》 在Java开发领域,Spring框架以其强大的功能和广泛的应用而备受开发者喜爱。...此外,掌握这些核心概念后,无论是使用Spring框架还是其他类似的IoC容器,都将更加得心应手。

    Spring IOC容器实现分析

    本文将基于Spring 2.0版本的源码,深入分析Spring IOC容器的内部实现机制,帮助读者更好地理解和使用这一强大的工具。 #### 二、核心概念与术语 在深入探讨Spring IOC容器之前,我们首先需要了解几个核心的概念: ...

Global site tag (gtag.js) - Google Analytics