写点什么

Spring Boot 精讲,看完你还敢说你不会 Spring Boot ?

用户头像
极客good
关注
发布于: 刚刚

// Do not initialize FactoryBeans here: We need to leave all regular beans


// uninitialized to let the bean factory post-processors apply to them!


// Separate between BeanDefinitionRegistryPostProcessors that implement


// PriorityOrdered, Ordered, and the rest.


List<BeanDefinitionRegistryPostProcessor> currentRegistryProcessors = new ArrayList<>();


First, invoke the BeanDefinitionRegistryPostProcessors that implement PriorityOrdered.


这里获取到的 beanName 不是刚刚的增强器而是常量


org.springframework.context.annotation.internalConfigurationAnnotationProcessor


并且下面这个循环会把


ConfigurationClassPostProcessor 加载出来。那么大家会问 ConfigurationClassPostProcessor 什么时候加载的呢?


createApplicationContext 这个方法还记得吧!就是在这里加载的当 Spring 选择了 Servlet 模式就会加载配置


CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME


就会作为 beanNamePut 进 beanFactory






First, invoke the BeanFactoryPostProcessors that implement PriorityOrdered.



Next, invoke the BeanFactoryPostProcessors that implement Ordered.



Finally, invoke all other BeanFactoryPostProcessors.



4.这里就是把启动类选出来并且开始 ComponentScan 的地方,自动装配也是这里完成的 @Import。



Parse each @Configuration class


这就是找 Component 的实现类



5.扫描 Component 和自动装配,这个也很重要,而且段代码有几个递归也是比较复杂,也是希望大家能够自己 debug 进去看看这到底是


【一线大厂Java面试题解析+核心总结学习笔记+最新架构讲解视频+实战项目源码讲义】
浏览器打开:qq.cn.hn/FTf 免费领取
复制代码


如何递归的。


Process any @ComponentScan annotations


这个为什么是数组呢?当你使用 @ComponentScans(value ={@ComponentScan("com.example.test"), @ComponentScan("com.example.test1")})


这里的长度就会变成 3



开始以启动类扫描,这就是为什么启动类永远在所有包的最外层,如果要扫描其他模块或者启动类以外的包就要 @ScanComponent 这个注解。


The config class is annotated with @ComponentScan -> perform the scan immediately


这里面就开始扫描 doScan 方法,只要是有 @Component 注解的都会把每个类的 Definition 放到 scannedBeanDefinitions 里面。



这里有事解析这个 Bean,跟刚刚的 parser.parse 一样,只是一个是数组一个是单个 Bean。


并且他会递归调用


doProcessConfigurationClass()方法,并且重新执行一次获取看看这个类有没有 @ComponentScan,并且继续扫描直至结束。


parse(bdCand.getBeanClassName(), holder.getBeanName());


Process any @Import annotations


其实每一个 parse 方法都会走到这里,把每个类的注解都会去循环一次,直至没有注解位置,会把 @Import 注解的类全部加载出来,这就是自动装配的原理。


这就是为什么我其他 jar 包的类可以给 Spring 管理 @Import 就是一个重点,把这个类导入到 BeanFactory 里面。


processImports(configClass, sourceClass, getImports(sourceClass), filter, true);


创建 Bean 实例


========


创建 bean 实例,这里的方法过于复杂用文章非常难解释,笔者这里就把大致的思路说下。


1.getBean 的时候先去 createBean 如果有就返回没有的话 doCreateBean。


2.当 doCreateBean 的时候就会触发 bean 的生命周期的各个接口。


3.其实笔者发现一个东西,ApplicationContextAware 并不算是 bean 生命周期的一环把,而是输入上下文的一环。


4.因为 ApplicationContextAware 其实是由添加了


AnnotationConfigServletWebServerApplicationContext 类所导致的。


5.创建 bean 的我会有下面的 uml 图让大家更能理解 Bean 是如何创建的。





关于 ApplicationContext


====================


实现 ApplicationContextAware 接口会调用 setApplicationContext 方法,而

用户头像

极客good

关注

还未添加个人签名 2021.03.18 加入

还未添加个人简介

评论

发布
暂无评论
Spring Boot 精讲,看完你还敢说你不会 Spring Boot ?