前言
SpringBoot框架是开发中的一大利器,其简化了spring的xml的配置,遵循了”约定大于配置“的原则,使用注解对常用的配置做默认配置,减少使用xml配置模式。SpringBoot为常用框架封装了大量的starter,比如spring-boot-starter-web会整合springmvc和内嵌的tomcat。SpringBoot在底层封装了默认的配置,修改配置在application.yml全局配置文件。如今在pom.xml文件中引用starter就可以使用这个框架,使用application.properties或者application.yml进行项目的全局配置。
那么问题来了,项目中SpringBoot是怎么帮我们自动配置的?那就说到接下来要讲的SpringBoot的自动配置原理。
一、@SpringBootApplication
@SpringBootApplication是核心注解,这个注解是整个SpringBoot项目的入口注解,是SpringBoot项目必不可少的注解。SpringBoot项目启动的时候加载的主配置类,主启动类。这个注解控制着SpringBoot自动配置原理的命脉。
看图片,HomeworkApplication作为入口类,入口类中有一个main方法,main方法中使用SpringApplication.run来启动整个应用。@SpringBootApplication是一个复合注解,进去看下。
稍微解释下以上注解:
@Target:表示@SpringBootApplication这个注解可以标记在哪儿,是标记在类上还是方法属性上 标记在类上面Element.Type。比如:@Autowired注解可以标记在方法上面,也可以标记在参数上面,也可写在属性上面。
@Retention:设置当注解标注的类以什么方式保留
@Document:java doc会生成注解信息
@Inheried:表示这个类是否被继承
接下来就是比较重要的3个注解:
1、@SpringBootConfiguration:SpringBoot的配置类,将此注解标注在某个类上表示这个是一个SpringBoot的配置类。 2、@EnableAutoConfiguration:开启自动配置功能,工程中需要配置的东西,SpringBoot会帮我们自动配置;@EnableAutoConfiguration告诉SpringBoot开启自动配置功能。这样自动配置才会生效。 3、@ComponentScan:开启包的扫描功能,相当于spring中.xml配置中的<context:component-scan>,并没有指定basepackage,如果没有指定,spring底层会自动扫描当前配置类所在的包。
二、重点@EnableAutoConfiguration:开启自动配置
点进去@EnableAutoConfiguration注解,发现这个注解也是个复合注解。符合直接中最最重要的是@Import注解, 通过导入AutoConfigurationImportSelector中的selectImports()方法。在方法getCandidateConfigurations中通过SpringFactoriesLoader.loadFactoryNames()扫描所有具有META-INF/spring.factories的jar包。 这个jar包中包含了所有需要自动配置的类。
selectImports()方法
通过getCandidateConfigurations这个方法以返回值集合的形式存储配置的信息。SpringFactoriesLoader.loadFactoryNames()扫描jar包在META-ING/spring.factory中。
在META-INF中有spring.factory的jar包
在spring.factory中有很多的xxxAutoConfuguration的自动配置的key,存储的形式是以key=value的形式添加到spring容器中。整个过程就是将类路径下"META-INF/spring.factories"里面配置的所有EnableAutoConfiguration的值加入到容器中。
小结:
总体来说,@EnableAutoConfiguration注解通过@SpringBootApplication被间接的标记在了Spring Boot的启动类上。在SpringApplication.run(...)的内部就会执行selectImports()方法,找到所有JavaConfig自动配置类的全限定名对应的class,然后将所有自动配置类加载到Spring容器中。
详细说,@Import(EnableAutoConfigurationImportSelector.class),借助EnableAutoConfigurationImportSelector,@EnableAutoConfiguration可以帮助SpringBoot应用将所有符合条件的@Configuration配置都加载到当前SpringBoot创建并使用的IOC容器:通过@Import(AutoConfigurationImportSelector.class)导入的配置功能,
AutoConfigurationImportSelector中的方法getCandidateConfigurations,得到待配置的class的类名集合,这个集合就是所有需要进行自动配置的类,通过META-INF/spring.factories中的的xxxAutoconfiguration就知道哪些类需要自动配置。
相关自动配置的生效:
自动配置类的生效是有条件的,条件是以某种注解的形式体现。下面是常见的注解条件:
@ConditionalOnBean:当容器里有指定的bean的条件下。
@ConditionalOnMissingBean:当容器里不存在指定bean的条件下。
@ConditionalOnClass:当类路径下有指定类的条件下。
@ConditionalOnMissingClass:当类路径下不存在指定类的条件下。
@ConditionalOnProperty:指定的属性是否有指定的值,比如@ConditionalOnProperties(prefix=”xxx.xxx”, value=”enable”, matchIfMissing=true),代表当xxx.xxx为enable时条件的布尔值为true,如果没有设置的情况下也为true。
在自动配置类中有相应的属性,通过@xxxProperities来封装配置文件中的相关属性。自动配置类xxxAutoConfiguration目的是给容器中添加组件等。
面试中我们可以这样回答:
SpringBoot有一个注解@SpringBootApplication,是一个复合或者派生注解。当SpringBoot调用run方法的时候,SpringBoot会通过@EnableAutoConfiguration注解找到方法getCandidateConfigurations找到所有需要自动配置类的集合,以List集合返回,自动配置类存储在META-INF/spring.factories配置文件中,文件中以AutoConfiguration命名结尾的向spring容器中添加组件,通过以Properties结尾命名的类中取得在全局配置文件中配置的属性。XxxxProperties和@ConfigurationProperties注解与全局配置文件中对应的属性进行绑定,所以我们在项目中全局配置文件application.yml中可以修改server.port :8081等等。
这就是SpringBoot的自动配置原理。