基本概念:
自动配置:用注解来对一些常规的配置做默认配置,简化xml配置内容,使你的项目能够快速运行。自动装配最明显的特征就是不用xml文件去做配置,而是用properties文件或者yml文件,简单地写一些坐标就能实现各类负载地配置。
为什么需要自动装配:无须各种的配置文件,无须各种繁杂的pom坐标,一个main方法,就能run起来了。
以下有一张原理图可以最直接地表达自动装配地实现原理,参看CSDN文章:
图1. 自动装配原理图
原理实现相关注解
@SrpingBootConfigration
点开这个注解内部
关键作用的注解是@EnableAutoConfiguration。而@EnableAutoConfiguration也是一个复合注解,它包含了@AutoConfigurationPackage和@Import({AutoConfigurationImportSelector.class})。
@Import 为spring的注解,用来导入配置类或者一些需要前置加载的类,通过进一步点击查看源代码,可以看到AutoConfigurationImportSelector.class实现了BeanClassLoader接口,通过getCandidateConfigurations去导入所有地配置文件。
@Import 注解导入的该类,实现了DeferredImportSelector接口,而DeferredImportSelector又实现了ImportSelector接口,因此该类实现了ImportSelector接口。通过@Import导入的该接口的实现类会执行重写的selectImports(AnnotationMetadata annotationMetadata)方法,该方法的返回值是String[]类型,即需要自动配置/导入Spring容器中的类的全限定名。
以上就是为什么我们写地配置类只需要加入一个@Configuration注解,Spring Boot就可以轻松地自己找到想要地目标文件。
我们重新捋一下整个原理的过程:
loadSpringFactories()方法加载所有jar包下的META-INF/spring.factories文件中的配置,并将其整理为Map,传递出去。loadFactoryNames()接收到loadSpringFactories()的调用结果,挑选出key为EnableAutoConfiguration的部分,即需要自动配置的部分,将其返回。getAutoConfigurationEntry()获得META-INF/spring.factories需要自动配置的部分,在根据@OnXXXCondition注解筛选排除部分不生效的自动配置,将其返回给selectImports()方法。点进去Map中的EnableAutoConfiguration,可以看到如下全为"xxxxAutoConfiguration"格式的字符串,这就是selectImports()方法返回的需要添加到Spring容器中管理的组件的全限定类名,作用类似于使用@Import引入了@Configuration注解的类,通过这种方式实现自动配置。
为什么有的自动配置不会生效
在加载自动配置类的时候,并不是将spring.factories的配置全部加载进来,而是通过@Conditional等注解的判断进行动态加载。@Conditional其实是spring底层注解,意思就是根据不同的条件,来进行自己不同的条件判断,如果满足指定的条件,那么配置类里边的配置才会生效。
常见的Conditional注解:
@ConditionalOnClass : classpath中存在该类时起效
@ConditionalOnMissingClass : classpath中不存在该类时起效
@ConditionalOnBean: DI容器中存在该类型Bean时起效
@ConditionalOnMissingBean : DI容器中不存在该类型Bean时起效
@ConditionalOnResource : 指定的文件存在时起效
@ConditionalOnJava : 指定的Java版本存在时起效
@ConditionalOnWebApplication : Web应用环境下起效
@ConditionalOnNotWebApplication : 非Web应用环境下起效