解析BeanDefinition

本文详细解析了Spring框架中Bean定义的加载过程,包括如何解析XML配置文件中的id和name属性,以及如何创建并填充BeanDefinition对象。进一步讨论了如何处理未指定beanName的情况,并展示了如何构建BeanDefinitionHolder。

首先从元素解析及信息提取开始,进入BeanDefinitionDelegate类的parseBeanDefinitionElement方法。

public BeanDefinitionHolder parseBeanDefinitionElement(Element ele) {
		return parseBeanDefinitionElement(ele, null);
	}
public BeanDefinitionHolder parseBeanDefinitionElement(Element ele, BeanDefinition containingBean) {
		String id = ele.getAttribute(ID_ATTRIBUTE);
		String nameAttr = ele.getAttribute(NAME_ATTRIBUTE);

		List<String> aliases = new ArrayList<String>();
		if (StringUtils.hasLength(nameAttr)) {
			String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr, MULTI_VALUE_ATTRIBUTE_DELIMITERS);
			aliases.addAll(Arrays.asList(nameArr));
		}

		String beanName = id;
		if (!StringUtils.hasText(beanName) && !aliases.isEmpty()) {
			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);
		}

		AbstractBeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName, containingBean);
		if (beanDefinition != null) {
			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;
	}

以上便是对默认标签解析的全过程了。对spring的解析犹如洋葱剥皮一样,一层一层的进行,现在只能看到对属性id以及name的解析。在开始对属性展开全面解析前,spring在外层又做了一个当前层的功能框架,在当前层完成的主要工作包括如下内容。

(1)提取元素中的id以及name属性。

(2)进一步解析其他所有属性并统一封装至GenericBeanDefinition类型的实例中。

(3)如果检测到bean没有指定的beanName,那么使用默认规则为此bean生成beanName。

(4)将获取到的信息封装到BeanDefinitionHolder的实例中。

进一步查看步骤(2)中对标签其他属性的解析过程。

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

`BeanDefinition` 是 Spring 框架中的一个核心接口,用于描述 Spring 容器中 Bean 的元数据信息。它定义了 Bean 的配置信息,包括 Bean 的类名、作用域、依赖关系、属性值等。Spring 容器通过解析 `BeanDefinition` 来创建和管理 Bean 实例。 ### 主要作用 1. **抽象 Bean 配置**:将 Bean 的定义(如类名、属性、依赖等)抽象为对象,便于 Spring 容器统一管理。 2. **支持多种配置方式**:无论是 XML 配置、注解配置还是 Java 配置,最终都会转换为 `BeanDefinition` 对象。 3. **动态修改 Bean 定义**:可以在运行时通过编程方式修改 `BeanDefinition`,实现动态 Bean 配置。 ### 核心属性 `BeanDefinition` 接口定义了许多属性,用于描述 Bean 的行为,常见的包括: - `beanClassName`:Bean 的类名。 - `scope`:Bean 的作用域(如 `singleton`、`prototype`)。 - `lazyInit`:是否延迟初始化。 - `dependsOn`:依赖的其他 Bean。 - `propertyValues`:Bean 的属性值。 - `constructorArgumentValues`:构造函数的参数值。 - `initMethodName`:初始化方法名。 - `destroyMethodName`:销毁方法名。 ### 实现类 Spring 提供了多个 `BeanDefinition` 的实现类,常见的有: 1. **`GenericBeanDefinition`**:通用实现,支持动态修改。 2. **`RootBeanDefinition`**:表示一个独立的 Bean 定义,可以合并父 Bean 定义。 3. **`ChildBeanDefinition`**:表示子 Bean 定义,继承父 Bean 定义的属性。 4. **`AnnotatedBeanDefinition`**:支持注解配置的 Bean 定义。 ### 示例代码 以下是一个通过编程方式创建 `GenericBeanDefinition` 的示例: ```java import org.springframework.beans.factory.support.GenericBeanDefinition; import org.springframework.beans.factory.support.BeanDefinitionRegistry; import org.springframework.beans.factory.support.DefaultListableBeanFactory; public class BeanDefinitionExample { public static void main(String[] args) { // 创建 BeanDefinitionRegistry(Spring 容器的一部分) BeanDefinitionRegistry registry = new DefaultListableBeanFactory(); // 创建 GenericBeanDefinition GenericBeanDefinition beanDefinition = new GenericBeanDefinition(); beanDefinition.setBeanClassName("com.example.MyBean"); beanDefinition.setScope("singleton"); // 注册 Bean 定义 registry.registerBeanDefinition("myBean", beanDefinition); // 从容器中获取 Bean MyBean myBean = (MyBean) registry.getBean("myBean"); System.out.println("Bean 获取成功: " + myBean); } } class MyBean { @Override public String toString() { return "This is MyBean"; } } ``` ### 工作流程 1. **解析配置**:Spring 容器在启动时解析配置文件(XML、注解或 Java 配置),将其转换为 `BeanDefinition` 对象。 2. **注册 Bean 定义**:将 `BeanDefinition` 注册到 `BeanDefinitionRegistry` 中。 3. **创建 Bean 实例**:Spring 容器根据 `BeanDefinition` 创建和管理 Bean 实例。 ### 动态修改 Bean 定义 通过 `BeanDefinitionRegistryPostProcessor` 可以动态修改 `BeanDefinition`,例如: ```java import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; import org.springframework.beans.factory.support.BeanDefinitionRegistry; import org.springframework.beans.factory.support.GenericBeanDefinition; import org.springframework.context.annotation.BeanDefinitionRegistryPostProcessor; public class CustomBeanDefinitionModifier implements BeanDefinitionRegistryPostProcessor { @Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) { // 修改已存在的 Bean 定义 if (registry.containsBeanDefinition("myBean")) { GenericBeanDefinition beanDefinition = (GenericBeanDefinition) registry.getBeanDefinition("myBean"); beanDefinition.setScope("prototype"); // 修改作用域 } } @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) { // 其他配置 } } ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值