Spring Bean生命周期之BeanDefinition的合并过程详解
写在前面
注:本文章使用的 SpringBoot 版本为 2.2.4.RELEASE,其 Spring 版本为 5.2.3.RELEASE
前言
书接上文,BeanDefinition注册到IoC容器后,紧接着就是要使用Bean了,要使用必须先要获取Bean,这里我们就以DefaultListableBeanFactory#getBean
方法来引出本次讨论的内容:BeanDefinition的合并
通过前面的章节我们了解到了BeanDefinition,那什么是BeanDefinition的合并呢?为什么要进行合并呢? 带着这个问题,我们到源码中去找找答案。
为了使源码逻辑有个参照,这里先给出一个案例,在分析源码时 将这个案例也代入进去方便我们理解源码
BeanDefinition的合并源码分析
实体类
@Data public class SuperUser implements Serializable { private String address; public SuperUser(String address) { this.address = address; } public SuperUser() { } } @Data @ToString(callSuper = true) public class User extends SuperUser { private String name; private Integer age; public User() { } public User(String name, Integer age) { this.name = name; this.age = age; } }
基于GenericBeanDefinition注册有层次的Bean
public class GenericBeanDefinitionDemo { public static void main(String[] args) { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); //父BeanDefinition GenericBeanDefinition rootBeanDefinition = new GenericBeanDefinition(); rootBeanDefinition.setBeanClass(SuperUser.class); //设置参数 MutablePropertyValues propertyValues = new MutablePropertyValues(); propertyValues.addPropertyValue("address", "地址"); rootBeanDefinition.setPropertyValues(propertyValues); //子BeanDefinition GenericBeanDefinition childBeanDefinition = new GenericBeanDefinition(); childBeanDefinition.setBeanClass(User.class); //设置构造参数 ConstructorArgumentValues argumentValues = new ConstructorArgumentValues(); argumentValues.addIndexedArgumentValue(0, "我就是我"); argumentValues.addIndexedArgumentValue(1, 18); childBeanDefinition.setConstructorArgumentValues(argumentValues); childBeanDefinition.setParentName("superUser"); //类型相同时 以子类为主 childBeanDefinition.setPrimary(true); context.registerBeanDefinition("superUser", rootBeanDefinition); context.registerBeanDefinition("user", childBeanDefinition); context.refresh(); User user = context.getBean("user", User.class); System.out.println(user); SuperUser superUser = context.getBean("superUser", SuperUser.class); System.out.println(superUser); context.close(); } }
在分析源码时我们要有侧重点,这里会将不太相关的逻辑一带而过。
AbstractBeanFactory#doGetBean
protected <T> T doGetBean(final String name, @Nullable final Class<T> requiredType, @Nullable final Object[] args, boolean typeCheckOnly) throws BeansException { //将name解析为beanName,如果传入的是alias,根据aliasMap进行转换,我们在前面介绍过了 final String beanName = transformedBeanName(name); Object bean; // 如果是单例Bean Object sharedInstance = getSingleton(beanName); if (sharedInstance != null && args == null) { //省略日志输出 // 这里的逻辑是根据beanName判断是否为FactoryBea,并采用相应方式去处理 bean = getObjectForBeanInstance(sharedInstance, name, beanName, null); } //如果不是单例对象 else { // 对原型对象进行验证,如果当前beanName已经在创建中了 抛出异常 if (isPrototypeCurrentlyInCreation(beanName)) { throw new BeanCurrentlyInCreationException(beanName); } // 获取父BeanFactory,前面我们介绍过了 BeanFactory允许有层级,可已存在父BeanFactory BeanFactory parentBeanFactory = getParentBeanFactory(); //如果存在父BeanFactory 去父BeanFactory中查找bean if (parentBeanFactory != null && !containsBeanDefinition(beanName)) { //省略去父BeanFactory查找Bean的过程 } //typeCheckOnly默认为false ,这里将beanName放到alreadyCreated集合中 表示该Bean正在创建中 if (!typeCheckOnly) { markBeanAsCreated(beanName); } try { // 这里来到了我们要重点关注的地方了,bd的合并 ⭐️ final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName); checkMergedBeanDefinition(mbd, beanName, args); //如果存在依赖Bean,需要进行依赖查找 String[] dependsOn = mbd.getDependsOn(); if (dependsOn != null) { // 省略dependsOn 依赖查找代码 } // 这里的if..else if .. else 是根据scope取值来的 //scope=singleton时 if (mbd.isSingleton()) { //省略单实例Bean创建过程 } //scope=prototype时 else if (mbd.isPrototype()) { //省略Prototype Bean创建过程 } //scope=request、application、session时 else { // 省略其他Scope Bean的创建过程 } if (requiredType != null && !requiredType.isInstance(bean)) { //省略类型转换代码 } // 返回创建的Bean return (T) bean; }
上面的方法实现比较长、比较复杂,这里只对重要的地方进行些注释说明并将与本次讨论无关的代码先行进行注释。
下面就进入到BeanDefinition的合并逻辑了
//假设beanName=user protected RootBeanDefinition getMergedLocalBeanDefinition(String beanName) throws BeansException { // 检查缓存,若存在直接返回 RootBeanDefinition mbd = this.mergedBeanDefinitions.get(beanName); if (mbd != null) { return mbd; } //getBeanDefinition(beanName)==>实际上去DefaultListableBeanFactory.beanDefinitionMap中根据key查找BeanDefinition,这在注册阶段已经放到beanDefinitionMap了。 return getMergedBeanDefinition(beanName, getBeanDefinition(beanName)); }
protected RootBeanDefinition getMergedBeanDefinition(String beanName, BeanDefinition bd) throws BeanDefinitionStoreException { return getMergedBeanDefinition(beanName, bd, null); } //根据上面的举例可知beanName=user,bd是User类的BeanDefinition,containingBd=null protected RootBeanDefinition getMergedBeanDefinition( String beanName, BeanDefinition bd, @Nullable BeanDefinition containingBd) throws BeanDefinitionStoreException { synchronized (this.mergedBeanDefinitions) { RootBeanDefinition mbd = null; // 尝试从缓存中拿 if (containingBd == null) { mbd = this.mergedBeanDefinitions.get(beanName); } if (mbd == null) { //如果当前BeanDefinition没有指定parentName,说明其不存在父BeanDefinition,不需要合并。以RootBeanDefinition形式展现 if (bd.getParentName() == null) { // 如果bd是RootBeanDefinition类型,直接类型转换 if (bd instanceof RootBeanDefinition) { mbd = ((RootBeanDefinition) bd).cloneBeanDefinition(); } else { //通过bd属性构造RootBeanDefinition mbd = new RootBeanDefinition(bd); } } else { // 走到这里说明存在parentName,当前bd需要与其父bd合并 BeanDefinition pbd; try { //得到父BeanName String parentBeanName = transformedBeanName(bd.getParentName()); //!beanName.equals(parentBeanName) 条件成立 说明当前beanName属于子bd if (!beanName.equals(parentBeanName)) { //递归地以父bd名称 查找父BeanDefinition。之所以递归地查找,是因为 可能此时的parentBeanName还有父,实体类存在多重继承关系 pbd = getMergedBeanDefinition(parentBeanName); } else { //走到这里,说明beanName.equals(parentBeanName),很有可能是父bd查找BeanDefinition时走来的。 //获取父BeanFactory,BeanFactory也是有层次的,有父子关系的,可参见ConfigurableBeanFactory#setParentBeanFactory BeanFactory parent = getParentBeanFactory(); if (parent instanceof ConfigurableBeanFactory) { pbd = ((ConfigurableBeanFactory) parent).getMergedBeanDefinition(parentBeanName); } else { throw new NoSuchBeanDefinitionException(parentBeanName, "Parent name '" + parentBeanName + "' is equal to bean name '" + beanName + "': cannot be resolved without an AbstractBeanFactory parent"); } } } catch (NoSuchBeanDefinitionException ex) { throw new BeanDefinitionStoreException(bd.getResourceDescription(), beanName, "Could not resolve parent bean definition '" + bd.getParentName() + "'", ex); } // pbd是父BeanDefinition,由其构造为RootBeanDefinition mbd = new RootBeanDefinition(pbd); //bd是子BeanDefinition,主要是继承父类的属性,并覆盖与父类同名的属性,有兴趣的可以看一下overrideFrom方法实现 mbd.overrideFrom(bd); } // 如果父bd未指定scope,则设置默认值 if (!StringUtils.hasLength(mbd.getScope())) { mbd.setScope(RootBeanDefinition.SCOPE_SINGLETON); } //由于containingBd=null 这里就不看了 if (containingBd != null && !containingBd.isSingleton() && mbd.isSingleton()) { mbd.setScope(containingBd.getScope()); } if (containingBd == null && isCacheBeanMetadata()) { this.mergedBeanDefinitions.put(beanName, mbd); } } //最终返回根据当前beanName找到的bd return mbd; } }
分析了上面的源码,我们试着总结一下:
1、如果不存在parentName,即不需要被合并,直接将bd转为RootBeanDefinition 返回即可
2、如果存在parentName
- 先根据parentName 找到父bd,若实体存在多级继承关系,则需要递归地查找。
- 将父bd转为RootBeanDefinition,并将子bd与父bd进行合并
- 设置一些其他属性
总结
本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!
相关文章
Spring Boot Maven 打包可执行Jar文件的实现方法
这篇文章主要介绍了Spring Boot Maven 打包可执行Jar文件的实现方法,需要的朋友可以参考下2018-02-02mybatis @InsertProvider报错问题及解决
这篇文章主要介绍了mybatis @InsertProvider报错的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教2022-07-07解决IDEA启动springboot项目报错java.lang.ClassNotFoundException: jav
这篇文章主要介绍了解决IDEA启动springboot项目报错java.lang.ClassNotFoundException: javax.servlet.ServletContext问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教2024-01-01java List.of()与Arrays.asList()方法对比分析
这篇文章主要为大家介绍了java List.of()与Arrays.asList()方法对比分析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪2023-11-11
最新评论