運(yùn)行原理
為了研究,我們正常從父項(xiàng)目的pom.xml開始進(jìn)行研究。
pom.xml
父依賴 spring-boot-starter-parent主要用來管理項(xiàng)目的資源過濾和插件
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.2.5.RELEASE</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
點(diǎn)父依賴進(jìn)去查看,發(fā)現(xiàn)還有一個(gè)父依賴spring-boot-dependencies,這里的這個(gè)父依賴才是真正管理springboot應(yīng)用里面的所有依賴版本的地方,是springboot的版本控制中心。
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.2.5.RELEASE</version>
<relativePath>../../spring-boot-dependencies</relativePath>
</parent>
啟動(dòng)器:spring-boot-starter-xxx:springboot的場景啟動(dòng)器
spring-boot-starter-web:導(dǎo)入web依賴的組件
<dependency> <groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
主程序
@SpringBootApplication
作用:標(biāo)注這是一個(gè)springboot主程序類,說明這是一個(gè)springboot應(yīng)用,springboot就是運(yùn)行這個(gè)類的mian方法啟動(dòng)的springboot應(yīng)用。
@SpringBootApplication //標(biāo)注這是一個(gè)主程序類,說明這是一個(gè)springboot應(yīng)用
public class Springboot01HelloworldApplication {
public static void main(String[] args) {
//這里啟動(dòng)了一個(gè)服務(wù),而不是執(zhí)行了一個(gè)方法。
SpringApplication.run(Springboot01HelloworldApplication.class, args);
}
}
點(diǎn)@SpringBootApplication繼續(xù)研究,會(huì)發(fā)現(xiàn)有@SpringBootConfiguration,@EnableAutoConfiguration,@ComponentScan這三個(gè)注解
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(
excludeFilters = {@Filter(
type = FilterType.CUSTOM,
classes = {TypeExcludeFilter.class}
), @Filter(
type = FilterType.CUSTOM,
classes = {AutoConfigurationExcludeFilter.class}
)}
)
1.@ComponentScan: spring自動(dòng)掃描包
這個(gè)我們?cè)趕pring配置文件中見到過,它用來自動(dòng)掃描并加載符合條件的組件或者bean,并將bean加載到IOC容器中。
2.@SpringBootConfiguration: springboot的配置類
標(biāo)注在某個(gè)類上,說明這個(gè)類是springboot的配置類,在這里它就說明SpringBootApplication這個(gè)類是springboot的配置類。
我們繼續(xù)點(diǎn)@SpringBootConfiguration進(jìn)去查看,會(huì)發(fā)現(xiàn) @Configuration這個(gè)注解
2.1 @Configuration:配置類,用來配置spring的xml文件
我們繼續(xù)點(diǎn)@Configuration進(jìn)去查看,會(huì)發(fā)現(xiàn) @Component這個(gè)注解。
2.2 @Component:組件,說明啟動(dòng)類本身也是一個(gè)組件,負(fù)責(zé)啟動(dòng)應(yīng)用。
至此,@SpringBootConfiguration這條線,我們研究完了。
3.@EnableAutoConfiguration:開啟自動(dòng)裝配,通過@EnableAutoConfiguration來幫我們自動(dòng)配置之前我們需要配置的東西。
我們繼續(xù)點(diǎn)@EnableAutoConfiguration進(jìn)去查看,會(huì)發(fā)現(xiàn) @AutoConfigurationPackage和@Import({AutoConfigurationImportSelector.class}) 這兩個(gè)注解。
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import({AutoConfigurationImportSelector.class})
public @interface EnableAutoConfiguration {
String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";
Class<?>[] exclude() default {};
String[] excludeName() default {};
}
3.1 @AutoConfigurationPackage自動(dòng)裝配包
繼續(xù)點(diǎn)進(jìn)去查看,出現(xiàn)@Import({Registrar.class})這個(gè)注解
3.1.1 @Import({Registrar.class}): spring底層注解,給容器導(dǎo)入一個(gè)組件
Registrar.class: 將主啟動(dòng)類所在包及所在包下面的所有子包里面所有的組件都掃描到Spring容器。
至此,@AutoConfigurationPackage這條線我們也研究完了。
3.2 @Import({AutoConfigurationImportSelector.class}): 給容器導(dǎo)入組件
AutoConfigurationImportSelector.class:自動(dòng)裝配導(dǎo)入選擇器。
導(dǎo)入的選擇器分析:
1.我們點(diǎn)進(jìn)去AutoConfigurationImportSelector.class這個(gè)類的源碼進(jìn)行探究,
2.我們點(diǎn)擊getCandidateConfigurations進(jìn)一步分析
protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
List<String> configurations = SpringFactoriesLoader.loadFactoryNames(this.getSpringFactoriesLoaderFactoryClass(), this.getBeanClassLoader());
Assert.notEmpty(configurations, "No auto configuration classes found in META-INF/spring.factories. If you are using a custom packaging, make sure that file is correct.");
return configurations;
}
2.1 使用了getSpringFactoriesLoaderFactoryClass()方法,返回一開始我們看到的啟動(dòng)自動(dòng)配置文件的注解類EnableAutoConfiguration.class
protected Class<?> getSpringFactoriesLoaderFactoryClass() {
return EnableAutoConfiguration.class;
}
2.2 發(fā)現(xiàn)它調(diào)用了SpringFactoriesLoader類的靜態(tài)方法,我們點(diǎn)擊loadFactoryNames進(jìn)入loadFactoryNames()
public static List<String> loadFactoryNames(Class<?> factoryType, @Nullable ClassLoader classLoader) {
ClassLoader classLoaderToUse = classLoader;
if (classLoader == null) {
classLoaderToUse = SpringFactoriesLoader.class.getClassLoader();
}
String factoryTypeName = factoryType.getName();
return (List)loadSpringFactories(classLoaderToUse).getOrDefault(factoryTypeName, Collections.emptyList());
}
發(fā)現(xiàn)它又調(diào)用了loadSpringFactories()方法,點(diǎn)進(jìn)去查看
private static Map<String, List<String>> loadSpringFactories(ClassLoader classLoader) {
Map<String, List<String>> result = (Map)cache.get(classLoader);
if (result != null) {
return result;
} else {
HashMap result = new HashMap();
try {
Enumeration urls = classLoader.getResources("META-INF/spring.factories");
while(urls.hasMoreElements()) {
URL url = (URL)urls.nextElement();
UrlResource resource = new UrlResource(url);
Properties properties = PropertiesLoaderUtils.loadProperties(resource);
Iterator var6 = properties.entrySet().iterator();
while(var6.hasNext()) {
Entry<?, ?> entry = (Entry)var6.next();
String factoryTypeName = ((String)entry.getKey()).trim();
String[] factoryImplementationNames = StringUtils.commaDelimitedListToStringArray((String)entry.getValue());
String[] var10 = factoryImplementationNames;
int var11 = factoryImplementationNames.length;
for(int var12 = 0; var12 < var11; ++var12) {
String factoryImplementationName = var10[var12];
((List)result.computeIfAbsent(factoryTypeName, (key) -> {
return new ArrayList();
})).add(factoryImplementationName.trim());
}
}
}
result.replaceAll((factoryType, implementations) -> {
return (List)implementations.stream().distinct().collect(Collectors.collectingAndThen(Collectors.toList(), Collections::unmodifiableList));
});
cache.put(classLoader, result);
return result;
} catch (IOException var14) {
throw new IllegalArgumentException("Unable to load factories from location [META-INF/spring.factories]", var14);
}
}
}
源碼分析:
- MultiValueMap<String, String> result = (MultiValueMap)cache.get(classLoader);獲得classLoader,我們返回可以看到這里得到的就是EnableAutoConfiguration標(biāo)注的類本身
- Enumeration<URL> urls = classLoader != null ? classLoader.getResources("META-INF/spring.factories") : ClassLoader.getSystemResources("META-INF/spring.factories");獲取一個(gè)資源 "META-INF/spring.factories"
- while循環(huán),讀取到的資源遍歷,封裝成為一個(gè)Properties
spring.factories文件
WebMvcAutoConfiguration
我們?cè)谏厦娴淖詣?dòng)配置類隨便找一個(gè)打開看看,比如 :WebMvcAutoConfiguration
都是大家熟悉的配置,所以,自動(dòng)配置真正實(shí)現(xiàn)是從classpath中搜尋所有的META-INF/spring.factories配置文件 ,并將其中對(duì)應(yīng)的 org.springframework.boot.autoconfigure. 包下的配置項(xiàng),通過反射實(shí)例化為對(duì)應(yīng)標(biāo)注了 @Configuration的JavaConfig形式的IOC容器配置類 , 然后將這些都匯總成為一個(gè)實(shí)例并加載到IOC容器中。
總結(jié)
- SpringBoot在啟動(dòng)的時(shí)候從類路徑下的META-INF/spring.factories中獲取EnableAutoConfiguration指定的值
- 將這些值作為自動(dòng)配置類導(dǎo)入容器,自動(dòng)配置類就生效,幫我們進(jìn)行自動(dòng)配置工作;
- 整個(gè)J2EE的整體解決方案和自動(dòng)配置都在springboot-autoconfigure的jar包中;
- 它會(huì)給容器中導(dǎo)入非常多的自動(dòng)配置類 (xxxAutoConfiguration), 就是給容器中導(dǎo)入這個(gè)場景需要的所有組件 , 并配置好這些組件 ;
- 有了自動(dòng)配置類 , 免去了我們手動(dòng)編寫配置注入功能組件等的工作;
主啟動(dòng)類
SpringApplication
@SpringBootApplication
public class SpringbootApplication {
public static void main(String[] args) { SpringApplication.run(SpringbootApplication.class, args);
}
}
分析:
- SpringbootApplication.class:應(yīng)用參數(shù)的入口
- args:命令行參數(shù)
- 該方法返回的是一個(gè)ConfigurableApplicationContext對(duì)象
SpringApplication主要做的事情:
- 推斷應(yīng)用的類型是普通的項(xiàng)目還是Web項(xiàng)目
- 查找并加載所有可用初始化器 , 設(shè)置到initializers屬性中
- 找出所有的應(yīng)用程序監(jiān)聽器,設(shè)置到listeners屬性中
- 推斷并設(shè)置main方法的定義類,找到運(yùn)行的主類
以上就是關(guān)于SpringBoot自動(dòng)裝配原理的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot框架的資料請(qǐng)關(guān)注W3Cschool其它相關(guān)文章!