热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

spring源码解析bean初始化与依赖注入四

2019独角兽企业重金招聘Python工程师标准前言本文转自“天河聊技术”微信公众号本次继续介绍spring源码解析bean初始化、依赖注入的部分正文上次介绍到这个方法org



2019独角兽企业重金招聘Python工程师标准>>> hot3.png



前言


本文转自“天河聊技术”微信公众号


本次继续介绍spring源码解析bean初始化、依赖注入的部分


 


正文


上次介绍到这个方法


org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#doCreateBean这一行


//     今早的缓存单例对象以便循环引用
     if (earlySingletonExposure) {
Object earlySingletonReference = getSingleton(beanName, false);
        if (earlySingletonReference != null) {
if (exposedObject == bean) {
exposedObject = earlySingletonReference;
           }
//          初始化的bean没被包装又有依赖
           else if (!this.allowRawInjectionDespiteWrapping && hasDependentBean(beanName)) {
//             获取依赖的bean
              String[] dependentBeans = getDependentBeans(beanName);
              Set actualDependentBeans &#61; new LinkedHashSet<>(dependentBeans.length);
              for (String dependentBean : dependentBeans) {
if (!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)) {
actualDependentBeans.add(dependentBean);
                 }
}
if (!actualDependentBeans.isEmpty()) {
throw new BeanCurrentlyInCreationException(beanName,
                       "Bean with name &#39;" &#43; beanName &#43; "&#39; has been injected into other beans [" &#43;
StringUtils.collectionToCommaDelimitedString(actualDependentBeans) &#43;
"] in its raw version as part of a circular reference, but has eventually been " &#43;
"wrapped. This means that said other beans do not use the final version of the " &#43;
"bean. This is often the result of over-eager type matching - consider using " &#43;
"&#39;getBeanNamesOfType&#39; with the &#39;allowEagerInit&#39; flag turned off, for example.");
              }
}
}
}

这一行


// Register bean as disposable.注册的bean作为可销毁的
try {
registerDisposableBeanIfNecessary(beanName, bean, mbd);
}

进入org.springframework.beans.factory.support.AbstractBeanFactory#registerDisposableBeanIfNecessary注册可销毁的bean


*/
  protected void registerDisposableBeanIfNecessary(String beanName, Object bean, RootBeanDefinition mbd) {
AccessControlContext acc &#61; (System.getSecurityManager() !&#61; null ? getAccessControlContext() : null);
//    不是原型模式&#xff0c;可以销毁
     if (!mbd.isPrototype() && requiresDestruction(bean, mbd)) {
//       如果是单例
        if (mbd.isSingleton()) {
// Register a DisposableBean implementation that performs all destruction
           // work for the given bean: DestructionAwareBeanPostProcessors,
           // DisposableBean interface, custom destroy method.
//          注册销毁的bean&#xff0c;这一部分在bean定义解析的部分介绍过&#xff0c;这里不作介绍了
           registerDisposableBean(beanName,
                 new DisposableBeanAdapter(bean, beanName, mbd, getBeanPostProcessors(), acc));
        }
else {
// A bean with a custom scope...
           Scope scope &#61; this.scopes.get(mbd.getScope());
           if (scope &#61;&#61; null) {
throw new IllegalStateException("No Scope registered for scope name &#39;" &#43; mbd.getScope() &#43; "&#39;");
           }
//          销毁后的回调方法
           scope.registerDestructionCallback(beanName,
                 new DisposableBeanAdapter(bean, beanName, mbd, getBeanPostProcessors(), acc));
        }
}
}

返回org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean这一行


try {
//                   bean初始化
                    return createBean(beanName, mbd, args);
                 }

这里介绍完了&#xff0c;往下


//                    发生异常销毁单例的bean
                    destroySingleton(beanName);

进入


org.springframework.beans.factory.support.DefaultSingletonBeanRegistry#destroySingleton


public void destroySingleton(String beanName) {
// Remove a registered singleton of the given name, if any.删除单例的bean&#xff0c;从本地缓存中删除
     removeSingleton(beanName);
     // Destroy the corresponding DisposableBean instance.
     DisposableBean disposableBean;
     synchronized (this.disposableBeans) {
//       从本地缓存中删除
        disposableBean &#61; (DisposableBean) this.disposableBeans.remove(beanName);
     }
//    bean销毁的逻辑
     destroyBean(beanName, disposableBean);
  }

这一行


从factoryBean中创建的bean对象
              bean &#61; getObjectForBeanInstance(sharedInstance, name, beanName, mbd);

这一行


else if (mbd.isPrototype()) {
//             bean定义为原型模式

这一行


      原型的bean创建之前维护内存中的缓存map映射信息
                 beforePrototypeCreation(beanName);
//                原型的bean创建
                 prototypeInstance &#61; createBean(beanName, mbd, args);

进入


org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#createBean(java.lang.String, org.springframework.beans.factory.support.RootBeanDefinition, java.lang.Object[])这一行


//        bean初始化之前执行beanProccessors
        Object bean &#61; resolveBeforeInstantiation(beanName, mbdToUse);

进入


org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#resolveBeforeInstantiation


&#64;Nullable
  protected Object resolveBeforeInstantiation(String beanName, RootBeanDefinition mbd) {
Object bean &#61; null;
     if (!Boolean.FALSE.equals(mbd.beforeInstantiationResolved)) {
// Make sure bean class is actually resolved at this point.
        if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {
Class targetType &#61; determineTargetType(beanName, mbd);
           if (targetType !&#61; null) {
//             调用bean初始化前处理器
              bean &#61; applyBeanPostProcessorsBeforeInstantiation(targetType, beanName);
              if (bean !&#61; null) {
//                bean初始化以后调用处理器
                 bean &#61; applyBeanPostProcessorsAfterInitialization(bean, beanName);
              }
}
}
mbd.beforeInstantiationResolved &#61; (bean !&#61; null);
     }
return bean;
  }

返回


org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#createBean(java.lang.String, org.springframework.beans.factory.support.RootBeanDefinition, java.lang.Object[])


这一行


//        bean创建
        Object beanInstance &#61; doCreateBean(beanName, mbdToUse, args);

返回


org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean


这一行


如果类型不一致进行类型转换
           T convertedBean &#61; getTypeConverter().convertIfNecessary(bean, requiredType);

返回


org.springframework.beans.factory.support.DefaultListableBeanFactory#preInstantiateSingletons


这一行


for (String beanName : beanNames) {
Object singletonInstance &#61; getSingleton(beanName);
        if (singletonInstance instanceof SmartInitializingSingleton) {
final SmartInitializingSingleton smartSingleton &#61; (SmartInitializingSingleton) singletonInstance;
           if (System.getSecurityManager() !&#61; null) {
AccessController.doPrivileged((PrivilegedAction) () -> {
//                单例bean创建完成后回调
                 smartSingleton.afterSingletonsInstantiated();
                 return null;
              }, getAccessControlContext());
           }
else {
smartSingleton.afterSingletonsInstantiated();
           }
}
}

bean初始化以后触发回调


这一行


单例bean创建完成后回调
                 smartSingleton.afterSingletonsInstantiated();

返回


org.springframework.context.support.AbstractApplicationContext#refresh


这一行 bean初始化、依赖注入


finishBeanFactoryInitialization(beanFactory);

 


最后


 


本次介绍到这里&#xff0c;以上内容仅供参考。







转载于:https://my.oschina.net/u/3775437/blog/1813308



推荐阅读
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 在重复造轮子的情况下用ProxyServlet反向代理来减少工作量
    像不少公司内部不同团队都会自己研发自己工具产品,当各个产品逐渐成熟,到达了一定的发展瓶颈,同时每个产品都有着自己的入口,用户 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • Linux重启网络命令实例及关机和重启示例教程
    本文介绍了Linux系统中重启网络命令的实例,以及使用不同方式关机和重启系统的示例教程。包括使用图形界面和控制台访问系统的方法,以及使用shutdown命令进行系统关机和重启的句法和用法。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • springmvc学习笔记(十):控制器业务方法中通过注解实现封装Javabean接收表单提交的数据
    本文介绍了在springmvc学习笔记系列的第十篇中,控制器的业务方法中如何通过注解实现封装Javabean来接收表单提交的数据。同时还讨论了当有多个注册表单且字段完全相同时,如何将其交给同一个控制器处理。 ... [详细]
  • 标题: ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 怀疑是每次都在新建文件,具体代码如下 ... [详细]
  • 利用Visual Basic开发SAP接口程序初探的方法与原理
    本文介绍了利用Visual Basic开发SAP接口程序的方法与原理,以及SAP R/3系统的特点和二次开发平台ABAP的使用。通过程序接口自动读取SAP R/3的数据表或视图,在外部进行处理和利用水晶报表等工具生成符合中国人习惯的报表样式。具体介绍了RFC调用的原理和模型,并强调本文主要不讨论SAP R/3函数的开发,而是针对使用SAP的公司的非ABAP开发人员提供了初步的接口程序开发指导。 ... [详细]
author-avatar
少钧13
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有