上一篇博客已经模拟springboot手写了一个简单springboot项目,相信大家对springboot有了一个基本的认识,设置更层次的理解,同时也简单的实现了springboot是如何通过删除和引入依赖就能做到tomcat和jetty的切换。今天继续聊聊springboot这一重要特性——自动装配
@ConditionalOnClass继上一篇博客,它就是判断某个类或者某些类是否存在,如果存在就会加载某个Bean,反之就不加载。比如判断应用内是否存在Tomcat.class,来决定是否使用tomcat来作为web服务器。
@ConditionalOnMissingBean
在springboot内部已经定义好了很多很多AutoConfiguration,比如AopAutoConfiguration,ServletWebServerFactoryAutoConfiguration等等,springboot在启动的时候就是通过条件注解来决定是否需要加载对应的对象解析为Bean对象。例如springboot启动默认使用的是tomcat,下面的ServletWebServerFactoryConfiguration.EmbeddedTomcat类肯定会被加载
此时我在启动类中定义一个bean:TomcatServletWebServerFactory
@Bean
public TomcatServletWebServerFactory tomcatServletWebServerFactory() {
return new TomcatServletWebServerFactory();
}
最终springboot只会使用程序员定义的TomcatServletWebServerFactory这个bean,而不会去加载springboot内部定义的bean。其实就是@ConditionalOnMissingBean(value = ServletWebServerFactory.class, search = SearchStrategy.CURRENT)这个注解起作用,它是判断某个是否不存在,如果不存在则会去加载springboot内部定义的,存在就表示程序员已经定义了这个bean,就不会去加载这个bean。具体是如何实现的呢?
?可以看到ConditionalOnMissingBean注解被@Conditional修饰了,spring会解析@Conditional传进去的OnBeanCondition类,因为ServletWebServerFactoryConfiguration.EmbeddedTomcat被@Configuration注解修饰了,所以当spring解析到ServletWebServerFactoryConfiguration.EmbeddedTomcat类时,会调用Condition#matches()方法,从而调用具体的OnBeanCondition#getMatchOutcome()方法判断当前spring容器是否存在TomcatServletWebServerFactory类型的bean,不存在则会继续解析EmbeddedTomcat,存在就不会去解析它了。
spring.factories
上篇博客使用SPI机制读取配置文件信息,实现无需修改代码引入maven依赖就可以切换使用tomcat或者jetty,而springboot框架真正使用的技术是将所有需要自动装配的类配置在spring.factories文件中,如图:
springboot启动时会创建spring容器并调用refreshContext(),这里会调用spring容器的refresh(),在此期间就会解析被@SpringBootApplication注解的主类,其中@EnableAutoConfiguration就是启动自动配置类,通过@Import(AutoConfigurationImportSelector.class)导入AutoConfigurationImportSelector,它实现了DeferredImportSelector接口,依次会调用process(),
该方法会读取这些文件并把它解析为Properties对象,最后分门别类封装到一个Map<String,List<String>>中,并缓存起来,最后会返回spring.factories中所有的AutoConfiguration。值得注意的是,springboot会解析所有项目的spring.factories,包括程序员所在的项目
然后通过new ArrayList<>(new LinkedHashSet<>(list))去重;随后获取需要排除的AutoConfiguration,包括@EnableAutoConfiguration中excludes属性和配置文件中配置的spring.autoconfigure.exclude;最后为了加快spring boot的启动速度,继续获取spring.factories中AutoConfigurationImportFilter配置的Condition(OnBeanCondition,OnClassCondition,OnWebApplicationCondition),它们的getOutcomes()会判断前面获取的AutoConfiguration是否符合条件,不符合条件的会被过滤掉,不会被spring解析。
?自动装配总结:
1、spring解析配置类时会解析@EnableAutoConfiguration注解,它会向spring容器导入AutoConfigurationImportSelector这个类,而它实现DeferredImportSelector,spring就会去执行process()方法,从而读取spring.factories所有的AutoConfiguration
2、使用LinkedHashSet去重
3、获取所有需要排除的AutoConfiguration(excludes属性和spring.autoconfigure.exclude)
4、获取spring.factories中AutoConfigurationImportFilter对AutoConfiguration进行过滤
|