在咱们的pom文件中最核心的依赖就一个:web
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.4.4</version>
<relativePath/>
</parent>
复制代码
它的父项目依赖,规定全部依赖的版本信息:spring
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.4.4</version>
</parent>
复制代码
由此,咱们发现springboot框架几乎声明了全部开发中经常使用的依赖的版本号,无需关注版本号,并且实现了自动版本仲裁机制,固然了咱们也能够根据咱们的须要,替换掉默认的依赖版本。springboot
@SpringBootApplication
public class BootApplication {
public static void main(String[] args) {
SpringApplication.run(BootApplication.class, args);
}
}
复制代码
在上面的启动类中咱们发现了一个陌生的注解@SpringBootApplication,这个注解的是什么含义呢?咱们点进去看一下。markdown
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = { @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
@Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
复制代码
其实@SpringBootApplication是上面三个注解的组合体,咱们对这三个注解理解清楚就能够了,下面逐个进行解释:app
@Configuration
public @interface SpringBootConfiguration {
复制代码
@Configuration咱们并不陌生,它容许在上下文中注册额外的bean或导入其余配置类,@SpringBootConfiguration其实表明当前类是一个配置类。框架
EnableAutoConfiguration的目的是启动SpringBoot的自动配置机制。ide
@AutoConfigurationPackage
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
复制代码
一、AutoConfigurationPackage指定默认的包规则spring-boot
@Import(AutoConfigurationPackages.Registrar.class)
public @interface AutoConfigurationPackage {
复制代码
AutoConfigurationPackage注解的做用是将 添加该注解的类所在的package做为 自动配置package进行管理。也就是说当SpringBoot应用启动时默认会将启动类所在的package做为自动配置的package。而后使用@Import注解将其注入到ioc容器中。这样,能够在容器中拿到该路径。this
static class Registrar implements ImportBeanDefinitionRegistrar, DeterminableImports {
@Override
public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) {
register(registry, new PackageImports(metadata).getPackageNames().toArray(new String[0]));
}
@Override
public Set<Object> determineImports(AnnotationMetadata metadata) {
return Collections.singleton(new PackageImports(metadata));
}
}
复制代码
重点看下registerBeanDefinitions方法。spa
方法的第二个参数经过new PackageImport(metadata).getPackageName()
方法设置。
接着看下PackageImport的构造器方法。
PackageImports(AnnotationMetadata metadata) {
AnnotationAttributes attributes = AnnotationAttributes
.fromMap(metadata.getAnnotationAttributes(AutoConfigurationPackage.class.getName(), false));
List<String> packageNames = new ArrayList<>(Arrays.asList(attributes.getStringArray("basePackages")));
for (Class<?> basePackageClass : attributes.getClassArray("basePackageClasses")) {
packageNames.add(basePackageClass.getPackage().getName());
}
if (packageNames.isEmpty()) {
packageNames.add(ClassUtils.getPackageName(metadata.getClassName()));
}
this.packageNames = Collections.unmodifiableList(packageNames);
}
复制代码
ClassUtils.getPackageName(metadata.getClassName())获取标注@AutoConfigurationPackage注解的类的全限定名。
最后,利用Registrar给容器中导入一系列组件,将指定的包下的全部组件导入进来。
二、@Import(AutoConfigurationImportSelector.class)
使用Import自动导入全部符合自动配置条件的Bean定义并加载到IOC容器
@Override
public void process(AnnotationMetadata annotationMetadata, DeferredImportSelector deferredImportSelector) {
Assert.state(deferredImportSelector instanceof AutoConfigurationImportSelector,
() -> String.format("Only %s implementations are supported, got %s",
AutoConfigurationImportSelector.class.getSimpleName(),
deferredImportSelector.getClass().getName()));
AutoConfigurationEntry autoConfigurationEntry = ((AutoConfigurationImportSelector) deferredImportSelector)
.getAutoConfigurationEntry(annotationMetadata);
this.autoConfigurationEntries.add(autoConfigurationEntry);
for (String importClassName : autoConfigurationEntry.getConfigurations()) {
this.entries.putIfAbsent(importClassName, annotationMetadata);
}
}
复制代码
一、利用getAutoConfigurationEntry(annotationMetadata);给容器中批量导入一些组件
二、调用List configurations = getCandidateConfigurations(annotationMetadata, attributes)获取到全部须要导入到容器中的配置类
三、利用工厂加载 Map<String, List> loadSpringFactories(@Nullable ClassLoader classLoader);获得全部的组件
四、从META-INF/spring.factories位置来加载一个文件。
默认扫描咱们当前系统里面全部META-INF/spring.factories位置的文件spring-boot-autoconfigure-2.4.4.RELEASE.jar包里面也有META-INF/spring.factories
文件里面写死了spring-boot一启动就要给容器中加载的全部配置类spring-boot-autoconfigure-2.4.4.RELEASE.jar/META-INF/spring.factories,一共130个自动配置类。
130个场景的全部自动配置,会在springboot启动的时候默认所有加载。xxxxAutoConfiguration会按照条件装配规则(@Conditional),最终会按需配置。
小结: SpringBoot为咱们的应用程序启用了三个功能:自动配置,组件扫描,以及可以在"应用类"上定义额外的配置。
@Component
在应用程序所在的软件包上启用扫描,指定扫描哪些Spring注解。
在130个场景有咱们比较熟悉两个组件,ServletWebServerFactoryAutoConfiguration和WebMvcAutoConfiguration,咱们以ServletWebServerFactoryAutoConfiguration为例,看一下SpringBoot是如何自动装配的webServer。
在注解中咱们看到了大量以@Conditional开头的注解,即条件装配,知足Conditional指定的条件,则进行组件注入。@EnableConfigurationProperties(ServerProperties.class)+@ConfigurationProperties(prefix = “server”, ignoreUnknownFields = true),读取咱们在配置文件编写的属性,并把它封装到JavaBean中,以供随时使用。
此时咱们的Tomcat容器已经以Bean的形式被注入到了IOC容器中。
若是发现应用中不须要特定自动配置类,则可使用exclude属性@SpringBootApplication
来禁用它们,如如下示例所示:
import org.springframework.boot.autoconfigure.*;
import org.springframework.boot.autoconfigure.jdbc.*;
@SpringBootApplication(exclude={DataSourceAutoConfiguration.class})
//@SpringBootApplication(excludeName = {"org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration"})
public class MyApplication {
}
复制代码
若是该类不在类路径中,则可使用excludeName
注释的属性,并指定彻底限定的名称(全类名字符串)。定义排除项,便可以是用哪一个注释级别也可使用属性来定义。
EnableAutoConfiguration —> 扫描xxxxxAutoConfiguration —> 根据条件@Conditional装配组件 —>根据xxxxProperties加载属性值 ---->application .properties