上一节将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
相关推荐
### Spring源码分析_Spring_IOC:深入理解Spring的IOC容器机制 #### 基本概念与核心作用 在探讨Spring框架的核心组件之一——IOC(Inversion of Control,控制反转)容器之前,首先需要理解它在Spring框架中的角色...
Spring 框架系列(7)- Spring IOC 实现原理详解之 IOC 初始化流程 本文将详细解释 Spring 框架中的 IOC...IOC 容器的初始化流程是 Spring 框架中的关键部分,用于将资源配置文件中的信息加载到 IOC 容器中。
3、源码分析-IOC容器的初始化 4、源码分析-IOC容器的依赖注入 5、源码分析-IOC容器的高级特性 三阶段 Spring AOP的涉及原理及具体实践 SpringJDBC的涉及原理及二次开发 SpringMVC框架设计原理及手写实现 四阶段 ...
《Spring IOC容器实现分析》 在Java开发领域,Spring框架无疑是使用最为广泛的轻量级框架之一,其中的核心组件就是IOC(Inversion of Control)容器。本文将深入剖析Spring的IOC容器,理解其工作原理和重要功能,以...
根据提供的文件信息,本次解读将围绕Spring框架的核心概念与源码分析进行展开。Spring框架作为Java企业级开发中不可或缺的一部分,其源码的学习对于深入理解框架机制、提高开发效率具有重要意义。下面,我们将从以下...
通过分析BeanFactory,我们可以了解对象的创建、初始化和依赖注入过程。 - **AOP**:面向切面编程允许我们定义横切关注点,如日志记录、事务管理,然后将其模块化为可重用的组件。 - **Data Access/...
文档可能将深入探讨Spring IoC容器初始化、Bean生命周期管理、依赖注入等关键概念,并以2021年的开源版本为背景进行分析。 从提供的部分文档内容来看,我们可以提炼出以下几个知识点: 1. **BeanFactory与...
Spring Core负责基础的IoC(Inversion of Control)容器,Spring Beans则实现了Bean的生命周期管理和配置,Spring Context则是基于Core之上构建的,提供了一个应用上下文,可以管理Bean并与其他Spring模块集成。...
2. **Spring Beans**:实现了IoC容器,通过XML或注解方式配置bean,并负责bean的初始化、装配和销毁。 3. **Spring AOP**:面向切面编程模块,允许定义方法拦截器和切入点,实现代码的解耦和模块化。 4. **Spring ...
5. **Bean管理**:最后,容器会管理Bean的生命周期,包括初始化、使用和销毁。 在实际开发中,Spring使用了更加高效和灵活的方式来实现这些功能,比如使用`ApplicationContext`接口作为IOC容器的入口,以及`...
1. **ApplicationContext**:这是Spring的核心接口,负责初始化、加载配置文件,管理Bean的生命周期。源码中可以看到它是如何解析XML或Java配置,创建并管理Bean的。 2. **BeanFactory**:作为ApplicationContext的...
通过阅读和学习这些源码,开发者可以了解到Spring如何实现IoC容器、AOP代理、事件机制、任务调度等多个关键功能。同时,这也有助于开发者更好地理解和使用Spring提供的API,以及在实际项目中如何定制和扩展Spring。 ...
1. **初始化流程**:从`org.springframework.context.support.ClassPathXmlApplicationContext`或`org.springframework.web.context.ContextLoader`开始,理解如何加载配置并创建Bean定义。 2. **依赖注入**:研究`...
在深入探讨Spring框架的核心组件和工作...在分析或运行Spring源码时,确保正确引入这两个库是确保项目正常运行的关键。通过深入了解这些工具库,开发者可以更好地理解和运用Spring框架,从而提高开发效率和应用性能。
IoC容器是Spring框架的心脏,它负责管理对象的生命周期和依赖关系,使得开发者能够实现松耦合和高可测试性的应用程序。 首先,我们来理解什么是IoC。IoC,也被称为依赖注入(Dependency Injection),是一种设计...
通过XML配置文件或Java注解,我们可以定义bean的创建、初始化和销毁过程,实现依赖注入,从而降低代码的耦合度。 2. AOP:Spring的AOP模块允许开发者定义横切关注点,如日志记录、事务管理等,将这些关注点与业务...
通过配置文件或者注解,开发者可以定义对象及其依赖关系,Spring容器会自动管理这些对象的生命周期,包括实例化、初始化、装配和销毁。 **Bean** 在Spring中,业务逻辑的组件被称为Bean。Bean是Spring容器管理的...
《简易Spring-ioc详解》 在Java开发领域,Spring框架以其强大的功能和广泛的应用而备受开发者喜爱。...此外,掌握这些核心概念后,无论是使用Spring框架还是其他类似的IoC容器,都将更加得心应手。
本文将基于Spring 2.0版本的源码,深入分析Spring IOC容器的内部实现机制,帮助读者更好地理解和使用这一强大的工具。 #### 二、核心概念与术语 在深入探讨Spring IOC容器之前,我们首先需要了解几个核心的概念: ...