我们通过获取bean,来看看如何 “合并BeanDefinition”,BeanDefinition在获取bean过程中的作用

“合并BeanDefinition” 的过程类似于面向对象中的继承,一个BeanDefinition 继承了另一个BeanDefinition(继承自己没有的,覆盖相同的)

通过bean标签的parent属性来指定“父BeanDefinition”

注意这里的继承仅仅是继承了Spring中BeanDefinition的属性,并不是bean对象的继承

1、测试用例

1.1、XML配置

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd"
>
<bean id="baseUser" class="com.yh.stu.spring.sourcestu._01ioc.ch1_base.mergebd.BaseUser">
</bean>
<bean id="user" class="com.yh.stu.spring.sourcestu._01ioc.ch1_base.mergebd.User" parent="baseUser"/>
</beans>

1.2、bean

public class BaseUser {
private Long id;
private String userName;
}

public class User {
private Long id;
private String addr;//地址
private String idCard;//身份证号
}

1.3、测试程序

public class MergeBeanDefinitionTest {
private static ApplicationContext ctx = new AnnotationConfigApplicationContext(Conf.class);
@Test
public void test(){
User user = ctx.getBean(User.class);
}
}

user bean定义时使用parent引用了其他的bean,在获取user的 BeanDefinition 时会 “合并BeanDefinition” 。

⭐️ BeanDefinition在获取bean过程中的作用
通过beanName获取 RootBeanDefinition ,根据 RootBeanDefinition 获取 是否singleton、是否prototype 决定如何创建bean

protected <T> T doGetBean(final String name, @Nullable final Class<T> requiredType,
@Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {
final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
String[] dependsOn = mbd.getDependsOn();// @DependsOn
if (dependsOn != null) {
...
}
if (mbd.isSingleton()) {
...
}else if (mbd.isPrototype()) {
...
}
}

从上面的程序可以看出 BeanDefinition 保存了bean的一些元信息,即是否单例、depentsOn数组等信息

2、调用栈

overrideFrom:290, AbstractBeanDefinition 
getMergedBeanDefinition:1253, AbstractBeanFactory
getMergedBeanDefinition:1192, AbstractBeanFactory
getMergedLocalBeanDefinition:1178, AbstractBeanFactory
doGetBean:251, AbstractBeanFactory
getBean:155, AbstractBeanFactory
preInstantiateSingletons:802, DefaultListableBeanFactory
finishBeanFactoryInitialization:874, AbstractApplicationContext
refresh:542, AbstractApplicationContext
<init>:92, AnnotationConfigApplicationContext
<clinit>:12, MergeBeanDefinitionTest

在org.springframework.beans.factory.support.AbstractBeanDefinition#overrideFrom 方法中,会使用User中的信息来覆盖BaseUser中的信息,类似面向对象中的继承。

public void overrideFrom(BeanDefinition other) {
if (StringUtils.hasLength(other.getBeanClassName())) {
setBeanClassName(other.getBeanClassName());
}
if (StringUtils.hasLength(other.getScope())) {// other
setScope(other.getScope());
}
....
}

这里的other就是 BaseUser 的 BeanDefinition