• IOC容器加载过程及Bean的生命周期和后置处理器


    SpringIOC 容器加载过程

    第一步:实例化化容器:AnnotationConfigApplicationContext

    1. @Configuration
    2. @ComponentScan("cn.zhe")
    3. public class MainStartTest {
    4. public static void main(String[] args) {
    5. // SpringIOC 出发点 加载Spring上下文
    6. AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainStartTest.class);
    7. HelloSpring bean = applicationContext.getBean(HelloSpring.class);
    8. bean.sayHello();
    9. }
    10. }

    构造函数

    1. // 根据参数可知,可以传入多个Class,但这种情况及其少见
    2. public AnnotationConfigApplicationContext(Class... componentClasses) {
    3. /**
    4. * 调用无参构造函数
    5. * 主要分三步:
    6. * a.调用父类构造函数
    7. * b.本类构造函数 初始化注解模式下的 bean定义读取器 AnnotatedBeanDefinitionReader
    8. * c.本类构造函数 初始化 classPath类型的 bean定义扫描器 AnnotatedBeanDefinitionScaner
    9. */
    10. this();
    11. /**
    12. * 注册配置类
    13. * 把传入的类进行注册,分为两种情况
    14. * a. @Configuration的配置类
    15. * b. 传入普通 Bean (基本不会这么做)
    16. * Spring把配置类分为两种
    17. * a. 带@Configuration注解的配置类称之为FULL配置类
    18. * b. 不带@Configuration注解,是带有@Component@Import@ImportResouce
    19. * @Service@ComponentScan等注解的配置类称之为Lite配置类
    20. */
    21. register(componentClasses);
    22. // 刷新IOC容器
    23. refresh();
    24. }


    this()方法分析开始

    第二步:实例化工厂:DefaultListableBeanFactory

    DefaultListableBeanFactory 就是我们所说的容器,里面放着beanDefinitionMap, beanDefinitionNames等

    1. // 调用无参构造,会先调用父类GenericApplicationContext的构造函数
    2. // 第一步调用父类构造函数,创建一个Bean工厂
    3. public GenericApplicationContext() {
    4. /**
    5. * 调用父类的构造函数,为 ApplicationContext spring 上下文对象初始 beanFactory
    6. * 因为 DefaultListableBeanFactory 是最底层的实现,功能是最全的
    7. */
    8. this.beanFactory = new DefaultListableBeanFactory();
    9. }
    10. // 第二三步
    11. public class AnnotationConfigApplicationContext extends GenericApplicationContext
    12. implements AnnotationConfigRegistry {
    13. // 注解bean定义读取器,主要作用是用来读取被注解的Bean
    14. private final AnnotatedBeanDefinitionReader reader;
    15. // 外部调用scan手动扫描的scanner对象,用处不大
    16. private final ClassPathBeanDefinitionScanner scanner;
    17. public AnnotationConfigApplicationContext() {
    18. /**
    19. * 初始化注解模式下的bean定义扫描器
    20. * 调用AnnotatedBeanDefinitionReader构造方法,传入的
    21. * 是this(AnnotationConfigApplicationContext)对象
    22. */
    23. this.reader = new AnnotatedBeanDefinitionReader(this);
    24. /**
    25. * 初始化classPath类型的bean定义扫描器
    26. * 使用场景极少,仅外部手动调用扫描使用,常规方式是不会用到scanner对象的
    27. * 此处扫描器仅用于自定义的扫描 applicationContext.scan();
    28. */
    29. this.scanner = new ClassPathBeanDefinitionScanner(this);
    30. }
    31. }

    第三步:实例化 BeanDefinition 读取器 (AnnotatedBeanDefinitionReader)

    主要就做了两件事情:

    • 注册内置 BeanPostProcessor
    • 注册内置相关核心的 BeanDefinition
    1. public AnnotatedBeanDefinitionReader(BeanDefinitionRegistry registry, Environment environment {
    2. Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
    3. Assert.notNull(environment, "Environment must not be null");
    4. // 把ApplicationContext对象赋值给AnnotatedBeanDefinitionReader
    5. this.registry = registry;
    6. // 用户处理条件表达式计算 @Condition
    7. this.conditionEvaluator = new ConditionEvaluator(registry, environment, null);
    8. // 注册一些配置的后置处理器,并注册Spring内置的多个Bean
    9. AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry);
    10. }

    关于registerAnnotationConfigProcessors(this.registry);方法内容较多,但大多相同的判断,注册 Spring 内置的多个 Bean,以ConfigurationClassPostProcesso为例:

    1. /**
    2. * 为容器中注册解析配置类的后置处理器 ConfigurationClassPostProcessor
    3. * org.springframework.context.annotation.internalConfigurationAnnotationProcessor
    4. */
    5. if (!registry.containsBeanDefinition(CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME)) {
    6. RootBeanDefinition def = new RootBeanDefinition(ConfigurationClassPostProcessor.class);
    7. def.setSource(source);
    8. beanDefs.add(registerPostProcessor(registry, def, CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME));
    9. }
    10. /**
    11. * 这方法为BeanDefinition设置了一个Role,ROLE_INFRASTRUCTURE代表这是spring内部的,并非用户定义的
    12. * registry.registerBeanDefinition(beanName, definition);是一个接口方法,
    13. * 实现类是 DefaultListableBeanFactory
    14. * 核心工作就是
    15. * a.this.beanDefinitionMap.put(beanName, beanDefinition);
    16. * 把beanName作为key,beanDefinition作为value,放到map里面
    17. * b.beanDefinitionNames就是一个List,这里就是把beanName放到List中去
    18. * DefaultListableBeanFactory就是我们所说的容器,里面放着beanDefinitionMap, beanDefinitionNames,
    19. * beanDefinitionMap是一个hashMap,beanName作为Key,beanDefinition作为Value,
    20. * beanDefinitionNames是一个集合,里面存放了beanName
    21. */
    22. private static BeanDefinitionHolder registerPostProcessor(
    23. BeanDefinitionRegistry registry, RootBeanDefinition definition, String beanName) {
    24. definition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
    25. // 核心已在上方注释
    26. registry.registerBeanDefinition(beanName, definition);
    27. return new BeanDefinitionHolder(definition, beanName);
    28. }

    逻辑就是:

    1. 判断容器中是否已经存在了ConfigurationClassPostProcessorBean
    2. 如果不存在,就通过 RootBeanDefinition 的构造方法获得ConfigurationClassPostProcessor的BeanDefinition
    3. 执行registerPostProcessor()方法,其内部就是注册Bean(与其他 Bean 注册流程一致)

    internalConfigurationAnnotationProcessor

    ConfigurationClassPostProcessor 是 Spring 中极其重要的一个类,它实现 BeanDefinitionRegistryPostProcessor 接口,BeanDefinitionRegistryPostProcessor 接口又扩展了 BeanFactoryPostProcessor 接口,BeanFactoryPostProcessor 是 Spring 的扩展点之一

    至此加载完以下扩展点(beanDefinition -> beanDefinitionMap)

    第四步:创建 BeanDefinition 扫描器 (ClassPathBeanDefinitionScanner)

    初始化 classPath 类型的 BeanDefinition 扫描器,使用场景极少,仅外部手动调用扫描使用,常规方式是不会用到 scanner 对象的此处扫描器仅用于自定义的扫描applicationContext.scan()

    至此this()方法结束


    register(annotatedClasses);分析开始

    第五步:注册配置类为BeanDefinition (register(annotatedClasses);)

    1. /**
    2. * 注册配置类
    3. * 把传入的类进行注册,分为两种情况
    4. * a. @Configuration的配置类
    5. * b. 传入普通 Bean (基本不会这么做)
    6. * Spring把配置类分为两种
    7. * a. 带@Configuration注解的配置类称之为FULL配置类
    8. * b. 不带@Configuration注解,是带有@Component@Import@ImportResouce
    9. * @Service@ComponentScan等注解的配置类称之为Lite配置类
    10. */
    11. register(componentClasses);
    12. public void register(Class... componentClasses) {
    13. this.reader.register(componentClasses);
    14. }
    1. private void doRegisterBean(Class beanClass, @Nullable String name,
    2. @Nullable Classextends Annotation>[] qualifiers, @Nullable Supplier supplier,
    3. @Nullable BeanDefinitionCustomizer[] customizers) {
    4. // AnnotatedGenericBeanDefinition可以理解为一种数据结构,是用来描述Bean的,这里的作用就是把传入
    5. // 的标记了注解的类转为AnnotatedGenericBeanDefinition数据结构,里面有一个getMetadata方法,可以拿到类上的注解
    6. AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(beanClass);
    7. // 判断是否需要跳过注解,spring中有一个@Condition注解,当不满足条件,这个bean就不会被解析
    8. if (this.conditionEvaluator.shouldSkip(abd.getMetadata())) {
    9. return;
    10. }
    11. abd.setInstanceSupplier(supplier);
    12. // 解析bean的作用域,如果没有设置的话,默认为单例
    13. ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(abd);
    14. abd.setScope(scopeMetadata.getScopeName());
    15. // 获得beanName
    16. String beanName = (name != null ? name : this.beanNameGenerator.generateBeanName(abd, this.registry));
    17. // 解析通用注解,填充到AnnotatedGenericBeanDefinition,
    18. // 解析的注解 Lazy,Primary,DependsOn,Role,Description
    19. AnnotationConfigUtils.processCommonDefinitionAnnotations(abd);
    20. if (qualifiers != null) {
    21. for (Classextends Annotation> qualifier : qualifiers) {
    22. if (Primary.class == qualifier) {
    23. abd.setPrimary(true);
    24. }
    25. else if (Lazy.class == qualifier) {
    26. abd.setLazyInit(true);
    27. }
    28. else {
    29. abd.addQualifier(new AutowireCandidateQualifier(qualifier));
    30. }
    31. }
    32. }
    33. if (customizers != null) {
    34. for (BeanDefinitionCustomizer customizer : customizers) {
    35. customizer.customize(abd);
    36. }
    37. }
    38. // 这个方法用处不大,就是把AnnotatedGenericBeanDefinition数据结构和beanName封装到一个对象中
    39. BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName);
    40. definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
    41. // 注册,最终会调用DefaultListableBeanFactory中的registerBeanDefinition方法去注册
    42. // DefaultListableBeanFactory维护着一系列信息,比如beanDefinitionNames,beanDefinitionMap
    43. // beanDefinitionMap是一个Map,用来保存beanName和beanDefinition
    44. BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, this.registry);
    45. }
    1. 通过AnnotatedGenericBeanDefinition的构造方法,获得配置类的BeanDefinition
    2. 判断需不需要跳过注册,Spring中有一个@Condition注解,如果不满足条件,就会跳过这个类的注册
    3. 然后是解析作用域,如果没有设置的话,默认为单例
    4. 获得BeanName
    5. 解析通用注解,填充到 AnnotatedGenericBeanDefinition,解析的注解为Lazy,Primary,DependsOn,Role,Description
    6. 限定符处理,不是特指@Qualifier注解,也有可能是Primary,或者是Lazy,或者是其他(理论上是任何注解,这里没有判断注解的有效性)
    7. 把AnnotatedGenericBeanDefinition数据结构和beanName封装到一个对象中(不重要,方便传参)
    8. 注册,最终会调用 DefaultListableBeanFactory 中的 registerBeanDefinition() 方法

    至此,注册配置类加载结束,配置类(MainStartTest,标记了@Configuration的类) 被放入 BeanDefinitionMap 中未实例化。(实例化都在reflesh()方法中进行)

    至此register()方法结束,将我们传入的配置类加载完毕即:mainStartTest


    refresh() 方法分析开始

    第六步:refresh();

    到这一步 Spring 还没有进行扫描,只是实例化了一个工厂,注册了一些内置的 Bean 和 配置类,这一行是至关重要的一个方法,也是内容最多的,里面做了大量的处理。

    1. @Override
    2. public void refresh() throws BeansException, IllegalStateException {
    3. synchronized (this.startupShutdownMonitor) {
    4. // 1:准备刷新上下文环境
    5. // 刷新预处理,和主流程关系不大,就是保存了容器的启动时间,启动标志等
    6. prepareRefresh();
    7. //2:告诉子类初始化Bean工厂(MVC),获取Bean工厂
    8. // 和主流程关系也不大,最终获得了DefaultListableBeanFactory
    9. ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
    10. // 3:对Bean工厂进行填充属性
    11. /**
    12. * ①添加了两个后置处理器:
    13. * a.ApplicationContextAwareProcessor
    14. * b.ApplicationListenerDetector
    15. * ②设置忽略自动装配和允许自动装配的接口,如果不存在某个bean的时候,
    16. * spring就自动注册singleton bean
    17. * ③ 设置了bean表达式解析器
    18. */
    19. prepareBeanFactory(beanFactory);
    20. try {
    21. // 4:空方法 留给子类去实现该接口 允许在上下文子类中对Bean工厂进行后置处理。
    22. postProcessBeanFactory(beanFactory);
    23. // 5:调用Bean工厂的后置处理器.
    24. // 执行自定义的BeanFactoryPostProcessor和内置的BeanFactoryPostProcessor
    25. invokeBeanFactoryPostProcessors(beanFactory);
    26. // 6:注册BeanPostProcessors
    27. registerBeanPostProcessors(beanFactory);
    28. // 7:初始化国际化资源处理器.
    29. initMessageSource();
    30. // 8:创建事件多播器
    31. initApplicationEventMulticaster();
    32. // 9:这个方法同样也是留个子类实现的springboot也是从这个方法进行启动tomcat的.
    33. // 模板方法,在容器刷新的时候可以自定义逻辑,不同的Spring容器做不同的事情
    34. onRefresh();
    35. // 10:将事件监听器注册到多播器上
    36. registerListeners();
    37. // 11:实例化懒加载单例Bean的,也就是Bean绝大部分都是在这里被创建出来的
    38. finishBeanFactoryInitialization(beanFactory);
    39. // 12:最后容器刷新 发布刷新事件(Spring cloud也是从这里启动的)
    40. finishRefresh();
    41. }
    42. catch (BeansException ex) {
    43. if (logger.isWarnEnabled()) {
    44. logger.warn("Exception encountered during context initialization - " +
    45. "cancelling refresh attempt: " + ex);
    46. }
    47. // Destroy already created singletons to avoid dangling resources.
    48. destroyBeans();
    49. // Reset 'active' flag.
    50. cancelRefresh(ex);
    51. throw ex;
    52. }
    53. finally {
    54. // 清除元数据缓冲,实例化后就不需要了
    55. resetCommonCaches();
    56. }
    57. }
    58. }

    下面来逐一解析里面比较重要的几个方法,有些不重要的就仅在上面注释了

    6.1 prepareBeanFactory(beanFactory)

    顾名思义,BeanFactory的一些准备工作

    1. 设置了一个类加载器
    2. 设置了bean表达式解析器
    3. 添加了属性编辑器的支持
    4. 添加了一个后置处理器:ApplicationContextAwareProcessor,此后置处理器实现了BeanPostProcessor接口
    5. 设置了一些忽略自动装配的接口
    6. 设置了一些允许自动装配的接口,并且进行了赋值操作
    7. 在容器中还没有XX的 bean 的时候,帮我们注册 beanName 为 XX 的 singleton bean

    6.2 invokeBeanFactoryPostProcessors(beanFactory)

    首先看一下我们到这一步时 BeanDefinitionMap 里面 bean 定义的情况:

    1. /**
    2. * 调用Bean工厂的后置处理器.
    3. * 执行自定义的 BeanFactoryPostProcessor 和内置的 BeanFactoryPostProcessor
    4. */
    5. protected void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory) {
    6. // getBeanFactoryPostProcessors(),获得外部可以手动添加一个后置处理器,如果不添加获得的集合永远为空
    7. PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors());
    8. // Detect a LoadTimeWeaver and prepare for weaving, if found in the meantime
    9. // (例如通过ConfigurationClassPostProcessor注册的@Bean方法)
    10. if (beanFactory.getTempClassLoader() == null && beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) {
    11. beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory));
    12. beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader()));
    13. }
    14. }
    1. public static void invokeBeanFactoryPostProcessors(
    2. ConfigurableListableBeanFactory beanFactory, List beanFactoryPostProcessors) {
    3. // 第一步:首先调用BeanDefinitionRegistryPostProcessor的后置处理器
    4. // 装beanName 后续会根据这个集合来判断处理器是否已经被执行过了
    5. Set<String> processedBeans = new HashSet<>();
    6. if (beanFactory instanceof BeanDefinitionRegistry) {
    7. // 强行把bean工厂转为BeanDefinitionRegistry
    8. BeanDefinitionRegistry registry = (BeanDefinitionRegistry) beanFactory;
    9. // 保存BeanFactoryPostProcessor类型的后置
    10. List regularPostProcessors = new ArrayList<>();
    11. List registryProcessors = new ArrayList<>();
    12. // 循环传递进来的 beanFactoryPostProcessors,正常情况为数据,只有手动添加了后置处理器才会有数据
    13. for (BeanFactoryPostProcessor postProcessor : beanFactoryPostProcessors) {
    14. // 判断后置处理器是不是 BeanDefinitionRegistryPostProcessor
    15. // 因为BeanDefinitionRegistryPostProcessor扩展了BeanFactoryPostProcessor
    16. if (postProcessor instanceof BeanDefinitionRegistryPostProcessor) {
    17. // 进行强制转换
    18. BeanDefinitionRegistryPostProcessor registryProcessor =
    19. (BeanDefinitionRegistryPostProcessor) postProcessor;
    20. // 调用作为BeanDefinitionRegistryPostProcessor的处理器的后置方法
    21. registryProcessor.postProcessBeanDefinitionRegistry(registry);
    22. // 添加到用于保存的BeanDefinitionRegistryPostProcessor的集合中
    23. registryProcessors.add(registryProcessor);
    24. }
    25. // 若没有实现BeanDefinitionRegistryPostProcessor 接口,那么它就是BeanFactoryPostProcessor
    26. // 把当前的后置处理器加入到regularPostProcessors中
    27. else {
    28. regularPostProcessors.add(postProcessor);
    29. }
    30. }
    31. // 定义一个集合用户保存当前准备创建的BeanDefinitionRegistryPostProcessor
    32. List currentRegistryProcessors = new ArrayList<>();
    33. // 第一步:去容器中获取BeanDefinitionRegistryPostProcessor的bean的处理器名称
    34. // internalConfigurationAnnotationProcessor即ConfigurationAnnotationProcessor
    35. String[] postProcessorNames =
    36. beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);
    37. // 循环上一步获取的BeanDefinitionRegistryPostProcessor的类型名称
    38. for (String ppName : postProcessorNames) {
    39. // 判断是否实现了PriorityOrdered接口的
    40. if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) {
    41. // 显示的调用getBean()的方式获取出该对象然后加入到currentRegistryProcessors集合中去
    42. currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));
    43. // 同时也加入到processedBeans集合中去
    44. // 后续会根据这个集合来判断处理器是否已经被执行过了
    45. processedBeans.add(ppName);
    46. }
    47. }
    48. // 对currentRegistryProcessors集合中BeanDefinitionRegistryPostProcessor进行排序
    49. sortPostProcessors(currentRegistryProcessors, beanFactory);
    50. // 把他加入到用于保存到registryProcessors中
    51. // 为什么要合并,因为registryProcessors是装载BeanDefinitionRegistryPostProcessor的
    52. // 一开始的时候,spring只会执行BeanDefinitionRegistryPostProcessor独有的方法
    53. // 而不会执行BeanDefinitionRegistryPostProcessor父类的方法,即BeanFactoryProcessor的方法
    54. // 所以这里需要把处理器放入一个集合中,后续统一执行父类的方法
    55. registryProcessors.addAll(currentRegistryProcessors);
    56. /**
    57. * 在这里典型的BeanDefinitionRegistryPostProcessor就是
    58. * ConfigurationClassPostProcessor
    59. * 用于进行bean定义的加载 比如我们的包扫描,@import 等等
    60. */
    61. // Spring热插播的体现,像ConfigurationClassPostProcessor就相当于一个组件
    62. // Spring很多事情就是交给组件去管理,如果不想用这个组件,直接去掉注册组件就行
    63. invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);
    64. // 调用完之后,马上clear掉,临时变量需要清除
    65. // list.clear()只清除对象的引用使其变为垃圾,与list = null 集合也会置空
    66. currentRegistryProcessors.clear();
    67. // 接下来,去容器中获取BeanDefinitionRegistryPostProcessor的bean的处理器名称
    68. postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);
    69. // 循环上一步获取的BeanDefinitionRegistryPostProcessor的类型名称
    70. for (String ppName : postProcessorNames) {
    71. // 没有被处理过,且实现了Ordered接口的
    72. if (!processedBeans.contains(ppName) && beanFactory.isTypeMatch(ppName, Ordered.class)) {
    73. // 显示的调用getBean()的方式获取出该对象然后加入到currentRegistryProcessors集合中去
    74. currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));
    75. // 同时也加入到processedBeans集合中去
    76. processedBeans.add(ppName);
    77. }
    78. }
    79. // 对currentRegistryProcessors集合中BeanDefinitionRegistryPostProcessor进行排序
    80. sortPostProcessors(currentRegistryProcessors, beanFactory);
    81. // 把他加入到用于保存到registryProcessors中
    82. registryProcessors.addAll(currentRegistryProcessors);
    83. // 调用他的后置处理方法
    84. invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);
    85. // 调用完之后,马上clear掉
    86. currentRegistryProcessors.clear();
    87. // 调用没有实现任何优先级接口的BeanDefinitionRegistryPostProcessor
    88. // 定义一个重复处理的开关变量 默认值为true
    89. // Finally, invoke all other BeanDefinitionRegistryPostProcessors until no further ones appear.
    90. boolean reiterate = true;
    91. // 第一次就可以进来
    92. while (reiterate) {
    93. // 进入循环马上把开关变量给改为fasle
    94. reiterate = false;
    95. // 去容器中获取BeanDefinitionRegistryPostProcessor的bean的处理器名称
    96. // 根据类型查 beanName 一般情况下只会获取到一个
    97. org.springframework.context.annotation.internalConfigurationAnnotationProcessor,
    98. 也就是 ConfigurationAnnotationProcessor
    99. postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);
    100. // 循环上一步获取的BeanDefinitionRegistryPostProcessor的类型名称
    101. for (String ppName : postProcessorNames) {
    102. // 没有被处理过的
    103. if (!processedBeans.contains(ppName)) {
    104. // 显示的调用getBean()的方式获取出该对象然后加入到currentRegistryProcessors集合中去
    105. currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));
    106. // 同时也加入到processedBeans集合中去
    107. processedBeans.add(ppName);
    108. // 再次设置为true
    109. reiterate = true;
    110. }
    111. }
    112. // 对currentRegistryProcessors集合中BeanDefinitionRegistryPostProcessor进行排序
    113. sortPostProcessors(currentRegistryProcessors, beanFactory);
    114. // 把他加入到用于保存到registryProcessors中
    115. registryProcessors.addAll(currentRegistryProcessors);
    116. // 调用他的后置处理方法
    117. invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);
    118. // 调用完之后,马上clear掉
    119. currentRegistryProcessors.clear();
    120. }
    121. // 调用实现了BeanDefinitionRegistryPostProcessor的接口 他是他也同时实现了BeanFactoryPostProcessor的方法
    122. invokeBeanFactoryPostProcessors(registryProcessors, beanFactory);
    123. // 调用BeanFactoryPostProcessor成品的不是通过getBean的
    124. invokeBeanFactoryPostProcessors(regularPostProcessors, beanFactory);
    125. }
    126. else {
    127. // 若当前的beanFactory没有实现了BeanDefinitionRegistry 直接调用
    128. // 直接调用 beanFactoryPostProcessor 接口的方法进行后置处理
    129. invokeBeanFactoryPostProcessors(beanFactoryPostProcessors, beanFactory);
    130. }
    131. // 获取容器中所有的 BeanFactoryPostProcessor
    132. String[] postProcessorNames =
    133. beanFactory.getBeanNamesForType(BeanFactoryPostProcessor.class, true, false);
    134. // 保存BeanFactoryPostProcessor类型实现了priorityOrdered
    135. List priorityOrderedPostProcessors = new ArrayList<>();
    136. // 保存BeanFactoryPostProcessor类型实现了Ordered接口的
    137. List<String> orderedPostProcessorNames = new ArrayList<>();
    138. // 保存BeanFactoryPostProcessor没有实现任何优先级接口的
    139. List<String> nonOrderedPostProcessorNames = new ArrayList<>();
    140. for (String ppName : postProcessorNames) {
    141. // processedBeans包含的话,表示在上面处理BeanDefinitionRegistryPostProcessor的时候处理过了
    142. if (processedBeans.contains(ppName)) {
    143. // skip - already processed in first phase above
    144. }
    145. // 判断是否实现了PriorityOrdered
    146. else if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) {
    147. priorityOrderedPostProcessors.add(beanFactory.getBean(ppName, BeanFactoryPostProcessor.class));
    148. }
    149. // 判断是否实现了Ordered
    150. else if (beanFactory.isTypeMatch(ppName, Ordered.class)) {
    151. orderedPostProcessorNames.add(ppName);
    152. }
    153. // 没有实现任何的优先级接口的
    154. else {
    155. nonOrderedPostProcessorNames.add(ppName);
    156. }
    157. }
    158. // 首先,先调用BeanFactoryPostProcessor实现了 PriorityOrdered接口的
    159. sortPostProcessors(priorityOrderedPostProcessors, beanFactory);
    160. invokeBeanFactoryPostProcessors(priorityOrderedPostProcessors, beanFactory);
    161. // 再调用BeanFactoryPostProcessor实现了 Ordered.
    162. List orderedPostProcessors = new ArrayList<>(orderedPostProcessorNames.size());
    163. for (String postProcessorName : orderedPostProcessorNames) {
    164. orderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class));
    165. }
    166. sortPostProcessors(orderedPostProcessors, beanFactory);
    167. invokeBeanFactoryPostProcessors(orderedPostProcessors, beanFactory);
    168. // 最后调用没有实现任何方法接口的
    169. List nonOrderedPostProcessors = new ArrayList<>(nonOrderedPostProcessorNames.size());
    170. for (String postProcessorName : nonOrderedPostProcessorNames) {
    171. nonOrderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class));
    172. }
    173. invokeBeanFactoryPostProcessors(nonOrderedPostProcessors, beanFactory);
    174. // Clear cached merged bean definitions since the post-processors might have
    175. // modified the original metadata, e.g. replacing placeholders in values...
    176. beanFactory.clearMetadataCache();
    177. }

    总结:

    首先,之前已经了解过 BeanDefinition 的两个扩展点 postProcessBeanFactory 和 postProcessBeanDefinitionRegistry。前者是可以修改 BeanDefinition,重写方法,后者可以多添加 BeanDefinition

    1、定义了一个 Set,processedBeans 装载BeanName,后面会根据此 Set 来判断后置处理器是否被执行过

    2、判断当前的 beanFactory 有没有实现 BeanDefinitionRegistry,当然是肯定的, 定义了两个 List 一个是 regularPostProcessors,用来装载 BeanFactoryPostProcessor。它只有一个实现 方法postProcessBeanFactory()。一个是 registryProcessors, 用来装载 BeanDefinitionRegistryPostProcessor。因为 它继承了 BeanFactoryPostProcessor 它不仅有postProcessBeanFactory()还有postProcessBeanDefinitionRegistry()

    3、循环传进来的beanFactoryPostProcessors,一般情况下都是空的,除非自己 add 了 beanFactory 的后置处理器。假设有数据,先判断是否是BeanDefinitionRegistryPostProcessor如果是调用postProcessBeanDefinitionRegistry()方法,并添加到集合 registryProcessors 中,否的话直接加入到集合 regularPostProcessors。(postProcessBeanFactory()会在后面执行,先存起来)

    4、 定义一个集合(List 临时变量) currentRegistryProcessors用户保存当前准备创建的 BeanDefinitionRegistryPostProcessor

    5、去容器中获取BeanDefinitionRegistryPostProcessor的bean的处理器名称internalConfigurationAnnotationProcessorConfigurationAnnotationProcessor一般情况都只会获取到一个。此时Spring还未扫描完成,扫描是在ConfigurationClassPostProcessor类完成的,就是下面的第一个invokeBeanDefinitionRegistryPostProcessors()方法

    6、循环 postProcessorNames internalConfigurationAnnotationProcessor判断是否实现了 PriorityOrdered,实现了添加到currentRegistryProcessors和processedBeans表示它们被处理过了(下一步才处理)

    7、对 currentRegistryProcessors 集合中BeanDefinitionRegistryPostProcessor 进行排序

    8、将currentRegistryProcessors集合加到registryProcessors集合中,因为registryProcessors是装载BeanDefinitionRegistryPostProcessor,一开始的时候,spring只会执行BeanDefinitionRegistryPostProcessor独有的方法postProcessBeanDefinitionRegistry()。而不会执行 BeanDefinitionRegistryPostProcessor 父类的方法,即 BeanFactoryProcessor 的方法postProcessBeanFactory()。所有在此统一放到一起等待后续执行

    9、internalConfigurationAnnotationProcessor(currentRegistryProcessors, registry),执行currentRegistryProcessors中的ConfigurationClassPostProcessor中的postProcessBeanDefinitionRegistry()方法,这里体现了 Spring 中热插拔,插件化开发的思想,如果不想用这个,不添加就行了。从下图可以看到,执行完该方法后 bean定义 被加载到了BeanDefinitionMap中

    10、清空currentRegistryProcessors,用完了就需要清空,给后面的其他的重复使用

    11、最后会重复上面的逻辑,调用顺序如下:

    • 实现了PriorityOrdered接口的
    • 实现了Ordered接口的
    • 没有实现任何的优先级接口的

    如果实现了多个的话,将在最先实现的地方调用,第二次将会判断是否已经处理过

    再来看一下postProcessBeanFactory()方法,它调用了一个会解析我们BeanDefinition的方法processConfigBeanDefinitions((BeanDefinitionRegistry) beanFactory);

    1. @Override
    2. public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) {
    3. int factoryId = System.identityHashCode(beanFactory);
    4. if (this.factoriesPostProcessed.contains(factoryId)) {
    5. throw new IllegalStateException(
    6. "postProcessBeanFactory already called on this post-processor against " + beanFactory);
    7. }
    8. this.factoriesPostProcessed.add(factoryId);
    9. if (!this.registriesPostProcessed.contains(factoryId)) {
    10. // BeanDefinitionRegistryPostProcessor hook apparently not supported...
    11. // Simply call processConfigurationClasses lazily at this point then.
    12. processConfigBeanDefinitions((BeanDefinitionRegistry) beanFactory);
    13. }
    14. // 为属性为full的Bean定义做CGLIB增强
    15. enhanceConfigurationClasses(beanFactory);
    16. beanFactory.addBeanPostProcessor(new ImportAwareBeanPostProcessor(beanFactory));
    17. }

    这个方法中会引申出一个知识点

    注册配置类 把传入的类进行注册,分为两种情况

    • a. @Configuration 的配置类
    • b. 传入普通 Bean (基本不会这么做)

    Spring 把配置类分为两种

    • a. 带 @Configuration 注解的配置类称之为 FULL 配置类
    • b. 不带 @Configuration 注解,而是带有 @Component,@Import,@ImportResouce, @Service, @ComponentScan 等注解的配置类称之为 Lite 配置类

    如果我们注册了Full 配置类,我们 getBean 这个配置类,会发现它已经不是原本那个配置类了,而是已经被 CGLIB 代理的类

    例如:写一个A类,其中有一个构造方法,打印出“HelloSpring”,再写一个配置类,里面有两个 带 @Bean 的方法。假设其中一个方法getA()new A(),并且返回A的对象。第二个方法又调用了getA()。如果配置类是 Lite 配置类,会发现打印了两次“HelloSpring”,即 A 类被 new 了两次。如果配置类是 FULL 配置类,会发现只打印一次,因为这个类被CGLIB代理了。

    1. public void processConfigBeanDefinitions(BeanDefinitionRegistry registry) {
    2. List configCandidates = new ArrayList<>();
    3. // 获取IOC 容器中目前所有bean定义的名称
    4. String[] candidateNames = registry.getBeanDefinitionNames();
    5. // 循环上一步获取的所有的Bean定义信息
    6. for (String beanName : candidateNames) {
    7. // 通过Bean的名称来获取我们的bean定义对象
    8. BeanDefinition beanDef = registry.getBeanDefinition(beanName);
    9. // 判断是否有没有解析过
    10. if (beanDef.getAttribute(ConfigurationClassUtils.CONFIGURATION_CLASS_ATTRIBUTE) != null) {
    11. if (logger.isDebugEnabled()) {
    12. logger.debug("Bean definition has already been processed as a configuration class: " + beanDef);
    13. }
    14. }
    15. // 进行正在的解析判断是不是完全的配置类 还是一个非正式的配置类
    16. else if (ConfigurationClassUtils.checkConfigurationClassCandidate(beanDef, this.metadataReaderFactory)) {
    17. // 满足添加就加入到候选的配置类集合中
    18. configCandidates.add(new BeanDefinitionHolder(beanDef, beanName));
    19. }
    20. }
    21. // Return immediately if no @Configuration classes were found
    22. // 若没有找到配置类直接返回
    23. if (configCandidates.isEmpty()) {
    24. return;
    25. }
    26. // Sort by previously determined @Order value, if applicable
    27. // 对配置类进行Order排序
    28. configCandidates.sort((bd1, bd2) -> {
    29. int i1 = ConfigurationClassUtils.getOrder(bd1.getBeanDefinition());
    30. int i2 = ConfigurationClassUtils.getOrder(bd2.getBeanDefinition());
    31. return Integer.compare(i1, i2);
    32. });
    33. // Detect any custom bean name generation strategy supplied through the enclosing application context
    34. // 创建我们通过@CompentScan导入进来的bean name的生成器
    35. // 创建我们通过@Import导入进来的bean的名称
    36. SingletonBeanRegistry sbr = null;
    37. if (registry instanceof SingletonBeanRegistry) {
    38. sbr = (SingletonBeanRegistry) registry;
    39. if (!this.localBeanNameGeneratorSet) {
    40. BeanNameGenerator generator = (BeanNameGenerator) sbr.getSingleton(
    41. AnnotationConfigUtils.CONFIGURATION_BEAN_NAME_GENERATOR);
    42. if (generator != null) {
    43. // 设置@CompentScan导入进来的bean的名称生成器
    44. this.componentScanBeanNameGenerator = generator;
    45. // 设置@Import导入进来的bean的名称生成器
    46. this.importBeanNameGenerator = generator;
    47. }
    48. }
    49. }
    50. if (this.environment == null) {
    51. this.environment = new StandardEnvironment();
    52. }
    53. // 创建一个配置类解析器对象
    54. // Parse each @Configuration class
    55. ConfigurationClassParser parser = new ConfigurationClassParser(
    56. this.metadataReaderFactory, this.problemReporter, this.environment,
    57. this.resourceLoader, this.componentScanBeanNameGenerator, registry);
    58. // 创建一个集合用于保存我们的配置类BeanDefinitionHolder集合默认长度是配置类集合的长度
    59. Set candidates = new LinkedHashSet<>(configCandidates);
    60. // 创建一个集合用于保存我们的已经解析的配置类,长度默认为解析出来默认的配置类的集合长度
    61. Set alreadyParsed = new HashSet<>(configCandidates.size());
    62. //do while 会进行第一次解析
    63. do {
    64. // 解析配置类
    65. // 经过这一步,会将@ComponentScans、@ComponentScan、@Bean、@Import等注解要注册的类扫描出来
    66. parser.parse(candidates);
    67. parser.validate();
    68. // 解析出来的配置类
    69. Set configClasses = new LinkedHashSet<>(parser.getConfigurationClasses());
    70. configClasses.removeAll(alreadyParsed);
    71. // Read the model and create bean definitions based on its content
    72. if (this.reader == null) {
    73. this.reader = new ConfigurationClassBeanDefinitionReader(
    74. registry, this.sourceExtractor, this.resourceLoader, this.environment,
    75. this.importBeanNameGenerator, parser.getImportRegistry());
    76. }
    77. // 把解析出来的配置类注册到容器中
    78. // 经过这一步会将@Bean、@import 注册的类变成BeanDefinition
    79. this.reader.loadBeanDefinitions(configClasses);
    80. // 加入到已经解析的集合中
    81. alreadyParsed.addAll(configClasses);
    82. candidates.clear();
    83. //判断我们IOC容器中的是不是>候选原始的bean定义的个数
    84. if (registry.getBeanDefinitionCount() > candidateNames.length) {
    85. // 获取所有的bean定义
    86. String[] newCandidateNames = registry.getBeanDefinitionNames();
    87. // 原始的老的候选的bean定义
    88. Set oldCandidateNames = new HashSet<>(Arrays.asList(candidateNames));
    89. Set alreadyParsedClasses = new HashSet<>();
    90. // 赋值已经解析的
    91. for (ConfigurationClass configurationClass : alreadyParsed) {
    92. alreadyParsedClasses.add(configurationClass.getMetadata().getClassName());
    93. }
    94. for (String candidateName : newCandidateNames) {
    95. // 表示当前循环的还没有被解析过
    96. if (!oldCandidateNames.contains(candidateName)) {
    97. BeanDefinition bd = registry.getBeanDefinition(candidateName);
    98. // 判断有没有被解析过
    99. // checkConfigurationClassCandidate 此时为Bean定义标识为full或lite,在后面根据属性潘森是否需要用CGLIB增强
    100. if (ConfigurationClassUtils.checkConfigurationClassCandidate(bd, this.metadataReaderFactory) &&
    101. !alreadyParsedClasses.contains(bd.getBeanClassName())) {
    102. candidates.add(new BeanDefinitionHolder(bd, candidateName));
    103. }
    104. }
    105. }
    106. candidateNames = newCandidateNames;
    107. }
    108. }
    109. // 存在没有解析过的 需要循环解析
    110. while (!candidates.isEmpty());
    111. // Register the ImportRegistry as a bean in order to support ImportAware @Configuration classes
    112. if (sbr != null && !sbr.containsSingleton(IMPORT_REGISTRY_BEAN_NAME)) {
    113. sbr.registerSingleton(IMPORT_REGISTRY_BEAN_NAME, parser.getImportRegistry());
    114. }
    115. if (this.metadataReaderFactory instanceof CachingMetadataReaderFactory) {
    116. // Clear cache in externally provided MetadataReaderFactory; this is a no-op
    117. // for a shared cache since it'll be cleared by the ApplicationContext.
    118. ((CachingMetadataReaderFactory) this.metadataReaderFactory).clearCache();
    119. }
    120. }

    至此自定义的配置类都加载到了beanDefinitonMap 中,但仍未初始化

    此处会实例化以下几个内置 Bean

    6.3 registerBeanPostProcessors(beanFactory);

    实例化和注册 beanFactory 中扩展了 BeanPostProcessor 的bean。

    例如:

    AutowiredAnnotationBeanPostProcessor(处理被@Autowired注解修饰的bean并注入)

    RequiredAnnotationBeanPostProcessor(处理被@Required注解修饰的方法)

    CommonAnnotationBeanPostProcessor(处理@PreDestroy、@PostConstruct、@Resource等多个注解的作用)等。

    此处会实例化以下几个内置 Bean

    6.4 initApplicationEventMulticaster(); 和 registerListeners();

    创建事件多播器

    注册监听器,广播early application events

    后续监听机制再来看这两个

    6.5 finishBeanFactoryInitialization(beanFactory);

    实例化 非懒加载单例 Bean ,也就是我们的 Bean 都是在这里被创建出来的。包括(实例化、填充属性、初始化)

    里面会有一个方法preInstantiateSingletons()是一个接口方法,这个方法只有一个实现类DefaultListableBeanFactory,里面最重要的就是getBean();。这中间还会去判断是否是一个特殊的Bean(即:FactoryBean)一旦一个类实现了 FactoryBean 并从写了getObject() 方法那么,IOC容器拿到的实例就是调用getObject方法得到的特殊的实例,没有实现这个接口时注册到IOC容器中的就是一个普通的Bean,当实现后,IOC容器会调用getObject方法返回的实例(工厂模式)

    1. // 初始化所有的非懒加载单例Bean
    2. beanFactory.preInstantiateSingletons();
    3. // 执行 getBean流程
    4. getBean(beanName);
    5. 复制代码

    里面调用的是 AbstractBeanFactory.java 里面的getBean();

    1. @Override
    2. public Object getBean(String name) throws BeansException {
    3. return doGetBean(name, null, null, false);
    4. }

    doGetBean 方法太多了,这里挑出主要创建 单例bean 的逻辑

    1. // 创建单例bean
    2. if (mbd.isSingleton()) {
    3. // 把beanName 和一个 singletonFactory 并且传入一个回调对象用于回调
    4. sharedInstance = getSingleton(beanName, () -> {
    5. try {
    6. // 进入创建bean的逻辑
    7. return createBean(beanName, mbd, args);
    8. }
    9. catch (BeansException ex) {
    10. // 创建bean的过程中发生异常,需要销毁关于当前bean的所有信息
    11. destroySingleton(beanName);
    12. throw ex;
    13. }
    14. });
    15. bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
    16. }

    这里的creatBean()又是一个接口方法,但也仅仅只有一个类对其做了实现AbstractAutowireCapableBeanFactory。该方法前面也会进行一大堆的判断,我们再次挑出关键步骤

    1. // 真正的开始创建Bean实例对象
    2. Object beanInstance = doCreateBean(beanName, mbdToUse, args);
    3. if (logger.isTraceEnabled()) {
    4. logger.trace("Finished creating instance of bean '" + beanName + "'");
    5. }
    6. return beanInstance;

    点进方法doCreateBean(beanName, mbdToUse, args);发现也在该类下面,里面又做了一大堆的事情,我们主要调出机构关键点

    创建实例

    1. // 使用合适的实例化策略来创建新的实例:工厂方法、构造函数自动注入、简单初始化
    2. instanceWrapper = createBeanInstance(beanName, mbd, args);

    填充属性及初始化

    1. // 给我们的属性进行赋值(调用set方法进行赋值)
    2. populateBean(beanName, mbd, instanceWrapper);
    3. // 进行对象初始化操作(在这里可能生成代理对象)
    4. exposedObject = initializeBean(beanName, exposedObject, mbd);

    initializeBean(beanName, exposedObject, mbd);中,又调用了invokeInitMethods(beanName, wrappedBean, mbd);invokeAwareMethods(beanName, bean);

    1. protected Object initializeBean(final String beanName, final Object bean, @Nullable RootBeanDefinition mbd) {
    2. if (System.getSecurityManager() != null) {
    3. AccessController.doPrivileged((PrivilegedAction<Object>) () -> {
    4. invokeAwareMethods(beanName, bean);
    5. return null;
    6. }, getAccessControlContext());
    7. }
    8. else {
    9. // 若我们的bean实现了XXXAware接口进行方法的回调
    10. invokeAwareMethods(beanName, bean);
    11. }
    12. Object wrappedBean = bean;
    13. if (mbd == null || !mbd.isSynthetic()) {
    14. // 调用我们的bean的后置处理器的postProcessorsBeforeInitialization方法 @PostCust注解的方法
    15. wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);
    16. }
    17. try {
    18. // 调用初始化方法
    19. invokeInitMethods(beanName, wrappedBean, mbd);
    20. }
    21. catch (Throwable ex) {
    22. throw new BeanCreationException(
    23. (mbd != null ? mbd.getResourceDescription() : null),
    24. beanName, "Invocation of init method failed", ex);
    25. }
    26. if (mbd == null || !mbd.isSynthetic()) {
    27. // 调用我们bean的后置处理器的PostProcessorsAfterInitialization方法
    28. wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);
    29. }
    30. return wrappedBean;
    31. }
    32. private void invokeAwareMethods(final String beanName, final Object bean) {
    33. if (bean instanceof Aware) {
    34. // 此bean实现了BeanNameAware
    35. if (bean instanceof BeanNameAware) {
    36. ((BeanNameAware) bean).setBeanName(beanName);
    37. }
    38. // 实现了BeanClassLoaderAware接口
    39. if (bean instanceof BeanClassLoaderAware) {
    40. ClassLoader bcl = getBeanClassLoader();
    41. if (bcl != null) {
    42. ((BeanClassLoaderAware) bean).setBeanClassLoader(bcl);
    43. }
    44. }
    45. // 实现了BeanFactoryAware
    46. if (bean instanceof BeanFactoryAware) {
    47. ((BeanFactoryAware) bean).setBeanFactory(AbstractAutowireCapableBeanFactory.this);
    48. }
    49. }
    50. }

    至此剩余的Bean也全部初始化完成至 IOC 容器中

    验证Spring Bean 的生命周期

    定义一个SpringBean

    1. @ComponentScan
    2. public class SpringBean
    3. implements InitializingBean, DisposableBean, BeanNameAware, BeanFactoryAware, BeanClassLoaderAware {
    4. // 就是一个普通的被@Component标注的类
    5. @Autowired
    6. AutoBean autoBean;
    7. public SpringBean() {
    8. System.out.println("SpringBean Constructor Method:" + autoBean);
    9. System.out.println("SpringBean()");
    10. }
    11. @Override
    12. public void setBeanClassLoader(ClassLoader classLoader) {
    13. System.out.println("ClassLoader");
    14. }
    15. @Override
    16. public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
    17. System.out.println("setBeanFactory");
    18. }
    19. @Override
    20. public void setBeanName(String name) {
    21. System.out.println("setBeanName:" + autoBean);
    22. System.out.println("setBeanName");
    23. }
    24. @Override
    25. public void destroy() throws Exception {
    26. System.out.println("destroy");
    27. }
    28. @Override
    29. public void afterPropertiesSet() throws Exception {
    30. System.out.println("afterPropertiesSet");
    31. }
    32. public void initMethod() {
    33. System.out.println("initMethod");
    34. }
    35. public void destroyMethod() {
    36. System.out.println("destroyMethod");
    37. }
    38. }

    再定义一个BeanPostProcessor,在重写的两个方法中进行了判断,如果传进来的 beanName 是 springBean 才进行打印

    1. @Component
    2. public class MyBeanPostProcessor implements BeanPostProcessor {
    3. @Override
    4. public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
    5. if(beanName.equals("springBean")) {
    6. System.out.println("postProcessBeforeInitialization");
    7. }
    8. return bean;
    9. }
    10. @Override
    11. public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
    12. if(beanName.equals("springBean")) {
    13. System.out.println("postProcessAfterInitialization");
    14. }
    15. return bean;
    16. }
    17. }

    定义一个配置类,完成自动扫描,但是SpringBean是手动注册的,并且声明了initMethod和destroyMethod:

    1. @Configuration
    2. @ComponentScan
    3. public class MainConfig {
    4. @Bean(initMethod = "initMethod",destroyMethod = "destroyMethod")
    5. public SpringBean springBean() {
    6. return new SpringBean();
    7. }
    8. }

    然后是启动类:

    1. public class Main {
    2. public static void main(String[] args) throws Exception {
    3. AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(MainConfig.class);
    4. // Spring 的 destroy() 方法已过时会报错,推荐使用 registerShutdownHook() 优雅的关闭 IOC
    5. // context.registerShutdownHook() 是一个钩子方法,当jvm关闭退出的时候会调用这个钩子方法
    6. // 当然 context.close() 也可以销毁容器
    7. context.registerShutdownHook();
    8. }
    9. }

    运行结果:

    由此我们可以推导出Spring的生命周期

    1. ① 实例化Bean对象,这个时候 Bean 的对象是非常低级的,基本不能够被我们使用,因为连最基本的属性都没有设置,可以理解为 连Autowired注解都是没有解析的
    2. ② 填充属性,当做完这一步,Bean对象基本是完整的了,可以理解为Autowired注解已经解析完毕,依赖注入完成了
    3. ③ 如果Bean实现了BeanNameAware接口,则调用setBeanName方法
    4. ④ 如果Bean实现了BeanClassLoaderAware接口,则调用setBeanClassLoader方法
    5. ⑤ 如果Bean实现了BeanFactoryAware接口,则调用setBeanFactory方法
    6. ⑥ 调用BeanPostProcessor的postProcessBeforeInitialization方法
    7. ⑦ 如果Bean实现了InitializingBean接口,调用afterPropertiesSet方法
    8. ⑧ 如果Bean定义了init-method方法,则调用Bean的init-method方法
    9. ⑨ 调用BeanPostProcessor的postProcessAfterInitialization方法。当进行到这一步,Bean已经被准备就绪了,一直停留在应用的上下文中,直到被销毁
    10. ⑩ 如果应用的上下文被销毁了,如果Bean实现了DisposableBean接口,则调用destroy方法,如果Bean定义了destory-method声明了销毁方法也会被调用

    思考

    配置类 @Configuration 加与不加的区别,加上会创建 CGLIB 动态代理,保证配置类中的 Bean 是受 IOC容器控制的,是单例的,如果配置类中重复的用某一个类,不加的话就是是重复调用方法,多次创建,不受 IOC 容器控制。

    重复 beanName 覆盖原则,如果是通过 Scanner 扫描到的同一包下两个相同的 beanName 会抛异常,如果一个是 @Compontent 扫描的 bean,一个是通过 @Bean 配置的 bean,那么 @Bean 配置的 bean 会覆盖前面的 bean,因为它是后创建的。

  • 相关阅读:
    Node.js浅学
    SpringCloud Alibaba(六) - Seata 分布式事务锁
    Vue2 Element DatePicker组件设置默认日期、控制日期范围
    C++语法基础(3)——分支结构程序设计
    【MySQL】(三)DDL数据库操作——数据库的创建、查看、修改、删除
    【实时语音转文本】PC端实时语音转文本(麦克风外音&系统内部音源)
    Docker Compose安装
    windows 的一些常见命令
    【跟小嘉学 Rust 编程】二十六、Rust的序列化解决方案(Serde)
    NanoDet-plus环境搭建过程中可能出现的BUG及解决方法
  • 原文地址:https://blog.csdn.net/LBWNB_Java/article/details/126279067