1.前言
前面几章 讲过 ClassPathXmlApplicationContext 是基于xml配置文件 来加载 spring 容器的:包括注册什么bean,扫描哪些包,事务,等等都是配置xml文件里,维护这个配置文件非常的困难。
所以就有了 基于注解的配置形式,由AnnotationConfigApplicationContext 上下文支持。需要以注解的配置形式来支持 xml配置文件所支持 的所有功能。
接下来我们通过源码来一一分析下。
2.AnnotationConfigApplicationContext 构造方法
public AnnotationConfigApplicationContext(String... basePackages) {
this();
scan(basePackages);
refresh();
}
接收的参数 为要 扫描的包 路径。
2.1 this()方法
第一行就是 this();
会初始化AnnotatedBeanDefinitionReader和ClassPathBeanDefinitionScanner属性。
2.1.1. 初始化AnnotatedBeanDefinitionReader
AnnotatedBeanDefinitionReader的构造方法中,会注册很多有用的BeanDefinitionRegistryPostProcessor类型的类,用来支持注解配置。
这个工作在基于xml配置解析中是用 在解析自定义标签 <conext:conpment-scan>的时候完成的,这里基于注解的 由构造方法里来注册。
这个方法会注册很多的BeanDefinitionRegistryPostProcessor类型的类,比较重要的是这几个
1. 注册 ConfigurationClassPostProcessor
这个类用来支持 @ComponentScan,@Bean,@Import等注解,非常重要
2.注册 AutowiredAnnotationBeanPostProcessor
这个类之前依赖注入的时候讲过了,用来支持@Autowired和@Value注解的
3.注册CommonAnnotationBeanPostProcessor
这个类之前依赖注入的时候讲过,用来支持@Resource依赖注入,以及方法上的@PostConstruct 和 @PreDestroy
2.1.2. 初始化ClassPathBeanDefinitionScanner
ClassPathBeanDefinitionScanner类 这个就是扫描器,基于包路径扫描,@ComponentScan 就是用的这个扫描器去扫描特定包下 的 含有@Component注解的类,并注册beanDefinition。
构造方法里的scan就要用他。
2.2. scan(basePackages);
AnnotationConfigApplicationContext的构造方法里要传入一个包路径,基于这个包路径 去扫描 需要注册的bean
这里就会用到 上一个方法 实例化的 ClassPathBeanDefinitionScanner 扫描器进行扫描。
2.2.1. 扫包
这个扫包的逻辑就是 递归的去扫,遍历每一个类文件,判断类上是否有 @Component注解,有的话返回
2.2.2. 判断是否需要注册
一个排除的过滤器集合,命中就 不需要注册,空的
一个 包含的过滤器集合,命中就可能需要注册,如果有@@Conditional,还需要判断@Conditional注解,逻辑在isConditionMatch方法中。
命中的过滤器,就是判断有没有 @Component注解,
2.3 refresh();
这个方法和ClassPathXmlApplicationContext 调的一样,都是父类的AbstractApplicationContext方法,是spring启动的核心方法。
3.总结
其实这个上下文 和 基于xml配置文件的上下文 殊途同归。差别最大的 是 ConfigurationClassPostProcessor,AutowiredAnnotationBeanPostProcessor ,CommonAnnotationBeanPostProcessor 这几个类的注册条件不一样,在基于注解的上下文里,这几个类 是默认需要注册的,它们所做的工作也没有差别。
由于xml配置的方式 在spring工程中已几近淘汰,后续其他功能的源码分析,我们就从AnnotationConfigApplicationContext的配置方式出发。