1. 为何引入一些SpringBoot依赖的时候,不需要指定版本?
总结:
spring-boot-dependencies:作为父工程,存放了SpringBoot的核心依赖。我们在写或者引入一些SpringBoot依赖的时候,不需要指定版本,正是因为SpringBoot的父依赖已经帮我们维护了一套版本。
细节:
SpringBoot的pom.xml
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.2.1.RELEASE</version> <relativePath/> <!-- lookup parent from repository --> </parent>
SpringBoot的项目都会存在一个父依赖,按住Ctrl+鼠标左键,可以点进去 --> 点进去之后发现里面除了一些插件和配置文件的格式之外,还存在一个依赖spring-boot-dependencies
于是再点进去,可以发现里面放了很多的依赖和依赖的版本号。
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-dependencies</artifactId> <version>2.2.1.RELEASE</version> <relativePath>../../spring-boot-dependencies</relativePath> </parent>
另外我们还可以看到,在父依赖中也帮我们写好了资源库,不用我们自己再去配置了。
<resources> <resource> <filtering>true</filtering> <directory>${basedir}/src/main/resources</directory> <includes> <!-- 可以读取的配置文件有 application.yml/application.yaml/application.properties --> <include>**/application*.yml</include> <include>**/application*.yaml</include> <include>**/application*.properties</include> </includes> </resource> <resource> <directory>${basedir}/src/main/resources</directory> <excludes> <exclude>**/application*.yml</exclude> <exclude>**/application*.yaml</exclude> <exclude>**/application*.properties</exclude> </excludes> </resource> </resources>
2. 启动器
启动器就是SpringBoot的启动场景,比如我们要使用web相关的,那么就直接引入spring-boor-starter-web,那么他就会帮我们自动导入web环境下所有必需的依赖。
SpringBoot会将所有的功能场景都封装成一个一个的启动器,供开发人员使用。
3. 【重点】自动装配原理 - SpringBoot主程序
文字版本(重要,必看):https://zhuanlan.zhihu.com/p/95217578
当我们的SpringBoot项目启动的时候,会先导入AutoConfigurationImportSelector,
这个类会帮我们选择所有候选的配置,我们需要导入的配置都是SpringBoot帮我们写好的一个一个的配置类,那么这些配置类的位置,存在与META-INF/spring.factories文件中。
通过这个文件,Spring可以找到这些配置类的位置,于是去加载其中的配置。
1-@SpringBootApplication注解,正是SpringBoot项目启动的核心。其包含:
- @SpringBootConfiguration//核心
- @EnableAutoConfiguration//核心
2-@SpringBootConfiguration其实就携带了一个@Configuration注解,这个注解我们再熟悉不过了,他就代表自己是一个Spring的配置类。所以我们可以认为:@SpringBootConfiguration = @Configuration
3-@EnableAutoConfiguration,顾名思义,这个注解一定和自动配置相关。其包含:
- @AutoConfigurationPackage //自动配置包
- @Import(AutoConfigurationImportSelector.class)//自动配置导入选择
4-@Import(AutoConfigurationImportSelector.class),帮我们导入了AutoConfigurationImportSelector,这个类中存在一个方法 getCandidateConfigurations() 可以帮我们从META-INF/spring.factories获取所有的配置。
5-META-INF/spring.factories
可以看到里面包含了很多自动配置属性:
6-我们可以随便找一个自动配置点进去,比如WebMvcAutoConfiguration
这里放了所有关于WebMvc的配置,如视图解析器、国际化等等。
@ConditionalOnXXX:如果其中的条件都满足,该类才会生效。
所以在加载自动配置类的时候,并不是将spring.factories的配置全量加载进来,而是通过这个注解的判断,如果注解中的类都存在,才会进行加载。
所以就实现了:我们在pom.xml文件中加入stater启动器,SpringBoot自动进行配置。完成开箱即用。
4. 【重点】spring.factories文件 与 yaml主配置文件 的关系
问题:我们在application.yaml主配置文件(详见:主配置文件applicaton.yaml解析)里写的配置,是如何作用于/override默认的配置呢?
答案:
xxxAutoConfiguration.class --> xxxProperties.class --> application.yaml
在第3节解释【springboot 自动装配流程图】中,有一个关键配置文件spring-boot-autoconfigure的META-INF/spring.factories。可以看到【spring.factories】里面的类都是以xxxAutoConfiguration结尾的。这里以HttpEncodingAutoConfiguration.java为例:
1) 原本,springboot通过【HttpEncodingAutoConfiguration.java】里@EnableConfigurationProperties(HttpProperties.class),会从HttpProperties.class读取默认配置
2) HttpProperties.class里还有个的 @ConfigurationProperties(prefix = "string.http")注解,通过这个注解与application.yaml链接了起来。只要在yaml中写入的相应的"string.http.xxx",就能override原有配置
【springboot 自动装配流程图】
【spring.factories】
【HttpEncodingAutoConfiguration.java】
通过【HttpEncodingAutoConfiguration.java】里@EnableConfigurationProperties(HttpProperties.class) 里的 @ConfigurationProperties(prefix = "string.http"),与application.yaml链接了起来
【HttpProperties.class】