生产中有很多形式的的配置方式,本文仅分析注解配置。对于其他形式的配置区别主观以为主要在配置文件的解析过程不同,不一一分析了。本文以利用Dubbo框架开发rpc服务端为例详细阐述配置类的解析、数据保存、实例化以及注入到容器中。

  通常,涉及到配置参数的框架类,基本都离不开配置的解析及配置的保存;至于配置的具体使用,各框架就自行处理了。对于Spring container而言,配置参数的具体使用就是在实例化Bean的时候使用。所以本文主要分两大部分:1、配置类的解析及数据保存;2、Bean的实例化并注入。

  具体示例配置如下:

    

    

  跟踪调试Spring的工作流程:

   

  查看这个类的属性:

      

  各属性的属性如下:

  构造函数首先调用父类GenericApplicationContext构造函数初始化BeanFactory,这个类中的属性beanDefinitionMap保存bean初始化需要的相关信息。

    

  其次本类中初始化reader和scanner, 分别用于记录BeanDefinition和扫描可能的Bean。初始化之后reader和scanner后,其内部属性如下:

    

  一、配置类的解析及数据保存(register)

    通过源码跟踪分析,此过程的最终目的就是将通用的注解配置以<key,value>的形式保存在DefaultListableBeanFactory的beanDefinitionMap中,使所有的bean全部暴露以便后续使用。该暴露非依赖循环,仅最外层可识别的所有bean。

  二、Bean的实例化并注入(refresh)

    beanDefinitionMap中有很多的bean定义,本文仅关注自定义的配置类providerConfiguration的实例化注入。在模板方法设计模式的文章中,以该方法阐述了实例化注册的大致执行流程。

    refresh方法中并没有传入beanFactory的参数,那么就得在此方法中获得beanFactory,具体子流程如下:

      

    进一步跟踪调试,prepareBeanFactory方法中对beanFactory相关属性进行框架本身的初始赋值(即与具体业务无关)。其中关于beanPostProcessor的如下:

      

      

      

    进入方法invokeBeanFactoryPostProcessors(beanFactory)在context中调用Bean工厂后置处理器依赖处理beanDefinitionMap中的bean定义,将其依赖的bean注入到beanDefinitionMap中。核心方法为processConfigBeanDefinitions(registry),通过该方法即可知其为处理配置bean定义。其处理核心逻辑为:

    1、将所有的配置类列出

      

    2、配置解析对象ConfigurationClassParser解析候选配置类保存至ConfigurationClass中:按层次递归处理配置类及其子类

      

    与示例配置类相关的解析注解有@PropertySource、@Import(@EnableDubbo中包含)、@Bean。属性相关的解析不做详细说明,主要阐述@Import/@Bean的解析。

    3、@Import与@Bean的处理

    在@Enable*注解的工作原理 - 池塘里洗澡的鸭子 - 博客园 (cnblogs.com)中,粗略涉及了@EnableDubbo。这里仅此分析@EnableDubbo中涉及动态注册Bean(导入与ImportBeanDefinitionRegistrar相关)处理。在解析器中,通过configClass.addImportBeanDefinitionRegistrar直接将ImportBeanDefinitionRegistrar作为配置类保存至ConfigClass中。

      

    对于dubbo自定义的注解也是@Import动态注册Bean的组合注解:

      

        

    同理,@Bean标记的Bean方法应该保存在beanMethos中。至此,与业务逻辑相关的bean定义全部保存到了configClasses中。最后通过loadBeanDefinitionsForConfigurationClass全部保存至beanDefinitionMap中。具体时序图如下:              

      

    接下来的处理才是真正的实例化处理:

    4、bean实例化

    非懒加载的实例化均在refresh#finishBeanFactoryInitialization中,跟踪调试大致流程如下:

      

    如果有代理实例化前利用beanPostProcessor将原有bean替换为目标对象返回,没有则直接利用反射实例化对象。关键部分如下:

      

      

      

      

      

    

Spring Bean配置加载为BeanDefinition全过程(注解配置)的更多相关文章

  1. Spring Bean 的加载过程

    Spring Bean 的加载过程 一个是populateBean,一个是initializeBean,这两个方法完成了bean的赋值与初始化. 这里有一个BeanDefinitionValueRes ...

  2. Spring Bean 的加载顺序

    一,单一Bean 装载 1. 实例化; 2. 设置属性值; 3. 如果实现了BeanNameAware接口,调用setBeanName设置Bean的ID或者Name; 4. 如果实现BeanFacto ...

  3. Spring Bean的加载

    Spring加载bean的原则:不等bean创建完成就会将创建bean的ObjectFactory提早曝光加入到缓存中.   单例bean在Spring容器里只会创建一次,后续创建时会首先从缓存中获取 ...

  4. BeanDefinitionLoader spring Bean的加载器

    spring 容器注册bean , 会把bean包装成beanDefinition 放进spring容器中,beanDefinitionLoader就是加载bean的类 . 一.源码 class Be ...

  5. spring bean容器加载后执行初始化处理@PostConstruct

    先说业务场景,我在系统启动后想要维护一个List常驻内存,因为我可能经常需要查询它,但它很少更新,而且数据量不大,明显符合缓存的特质,但我又不像引入第三方缓存.现在的问题是,该List的内容是从数据库 ...

  6. 1. Spring基于xml加载和读取properties文件配置

    在src目录下,新建test.properties配置文件,内容如下 name=root password=123456 logArchiveCron=0/5 * * * * ? 一种是使用sprin ...

  7. Java实现配置加载机制

    前言 现如今几乎大多数Java应用,例如我们耳熟能详的tomcat, struts2, netty…等等数都数不过来的软件,要满足通用性,都会提供配置文件供使用者定制功能. 甚至有一些例如Netty这 ...

  8. Spring源码加载BeanDefinition过程

    本文主要讲解Spring加载xml配置文件的方式,跟踪加载BeanDefinition的全过程. 源码分析 源码的入口 ClassPathXmlApplicationContext构造函数 new C ...

  9. 【Spring】详解Spring中Bean的加载

    之前写过bean的解析,这篇来讲讲bean的加载,加载要比bean的解析复杂些,该文之前在小编原文中有发表过,要看原文的可以直接点击原文查看,从之前的例子开始,Spring中加载一个bean的方式: ...

随机推荐

  1. 关于 cannot create Parameters: [] 报错问题的解决方法

    其实在Sort类中添加无参构造就可以解决 我自己写的是Sort类,其它情况得视你们自己写的类决定 至于为什么也不是很清楚

  2. 论文翻译:2020_RESIDUAL ACOUSTIC ECHO SUPPRESSION BASED ON EFFICIENT MULTI-TASK CONVOLUTIONAL NEURAL NETWORK

    论文翻译:https://arxiv.53yu.com/abs/2009.13931 基于高效多任务卷积神经网络的残余回声抑制 摘要 在语音通信系统中,回声会降低用户体验,需要对其进行彻底抑制.提出了 ...

  3. Mysql存储过程二

    1.MySQL中创建存储过程时通过DEFINER和SQL SECURITY设置访问权限 procedure与function.trigger等创建时紧接着CREATE都有个definer可选项,该de ...

  4. Metasploit生成木马入侵安卓手机

    开始 首先你需要一个Metasploit(废话) Linux: sudo apt install metasploit-framework Termux: 看这里 指令 sudo su //生成木马文 ...

  5. 《剑指offer》面试题53 - I. 在排序数组中查找数字 I

    问题描述 统计一个数字在排序数组中出现的次数. 示例 1: 输入: nums = [5,7,7,8,8,10], target = 8 输出: 2 示例 2: 输入: nums = [5,7,7,8, ...

  6. ctfshow萌新 web1-7

    ctfshow萌新 web1 1.手动注入.需要绕过函数inval,要求id不能大于999且id=1000,所以用'1000'字符代替数字1000 2.找到?id=" "处有回显 ...

  7. 【刷题-LeetCode】208. Implement Trie (Prefix Tree)

    Implement Trie (Prefix Tree) Implement a trie with insert, search, and startsWith methods. Example: ...

  8. 豆瓣爬虫——通过json接口获取数据

    最近在复习resqusts 爬虫模块,就重新写了一个豆瓣爬虫,这个网页从HTML 源码上来看是没有任何我想要的信息的,如下图所示: 这是网页视图,我在源码中查找影片信息,没有任何信息,如图: 由此我判 ...

  9. VUE3 之 插槽的使用 - 这个系列的教程通俗易懂,适合新手

    1. 概述 非理性定律告诉我们: 人们总是习惯于以情感去判断眼前的事物,非理性的去做决定. 对于长远的利益,人们更愿意去选择短期的利益. 因此在做决定前要让自己冷静,理性的分析,让自己看的更远. 言归 ...

  10. 深入学习python内存管理

    深入Python的内存管理   作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 语言的内存管理是语言设计的一个重要方面.它是决定语 ...