什么是SpringBoot?
回答這個(gè)問(wèn)題之前枢冤,我先回答一下什么是Spring鸠姨?
Spring框架是一個(gè)開(kāi)放源代碼的應(yīng)用框架,是針對(duì)Bean的生命周期進(jìn)行管理的輕量級(jí)容器淹真。
Spring是為了解決企業(yè)級(jí)應(yīng)用開(kāi)發(fā)的復(fù)雜性而創(chuàng)建的讶迁,主要的目的是“簡(jiǎn)化開(kāi)發(fā)“。
Spring如何簡(jiǎn)化開(kāi)發(fā)核蘸?
基于POJO的輕量級(jí)和最小入侵式編程巍糯。
通過(guò)IOC/DI和面向接口實(shí)現(xiàn)松耦合。
基于切面AOP策略和模板方法進(jìn)行聲明式編程客扎。
通過(guò)切面和模板減少樣式代碼祟峦。
SpringBoot的出現(xiàn)就是進(jìn)一步完善Spring的目的”簡(jiǎn)化開(kāi)發(fā)“,提出了約定優(yōu)于配置的思想徙鱼,通過(guò)“自動(dòng)裝配”宅楞,實(shí)現(xiàn)了進(jìn)一步的"簡(jiǎn)化開(kāi)發(fā)"。
SpringBoot如何實(shí)現(xiàn)自動(dòng)裝配袱吆?
Spring實(shí)現(xiàn)配置的方式厌衙。可以分為三種:JavaConfig绞绒、XML婶希、注解。
SpringBoot實(shí)現(xiàn)自動(dòng)裝配的方式是注解+JavaConfig处铛。
SpringBoot啟動(dòng)類(lèi)注解 @SpringBootApplication :
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration // springboot的核心配置
@EnableAutoConfiguration // 自動(dòng)裝配默認(rèn)啟用
@ComponentScan(
excludeFilters = {@Filter(
type = FilterType.CUSTOM,
classes = {TypeExcludeFilter.class}
), @Filter(
type = FilterType.CUSTOM,
classes = {AutoConfigurationExcludeFilter.class}
)}
)
public @interface SpringBootApplication {
//... 部分代碼可自行查看源碼
}
@EnableAutoConfiguration 注解內(nèi)部的代碼如下:
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage // 自動(dòng)裝配包
@Import({AutoConfigurationImportSelector.class}) // 導(dǎo)入 自動(dòng)裝配選擇器類(lèi) 由此深入了解
public @interface EnableAutoConfiguration {
String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";
Class<?>[] exclude() default {};
String[] excludeName() default {};
}
AutoConfigurationImportSelector.class 部分代碼:
/**
AutoConfigurationImportSelector.class 部分代碼
簡(jiǎn)單剖析
*/
// 獲得所有候選配置類(lèi)的路徑
protected List<String> getCandidateConfigurations(AnnotationMetadata metadata,
AnnotationAttributes attributes) {
// 找到META-INF/spring.factories中的配置類(lèi)的路徑
List<String> configurations = SpringFactoriesLoader
.loadFactoryNames(this.getSpringFactoriesLoaderFactoryClass(), this.getBeanClassLoader());
// 斷言一下這個(gè)list是否為空饲趋,如果為空就報(bào)錯(cuò)
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;
}
SpringFactoriesLoader.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();
// 核心是這個(gè)方法
return (List)loadSpringFactories(classLoaderToUse).getOrDefault(factoryTypeName, Collections.emptyList());
}
loadSpringFactories();
// 這個(gè)方法就是把SpringBoot所有存在的自動(dòng)裝配的配置類(lèi)的路徑返回
// 同時(shí)存儲(chǔ)到SpringFactoriesLoader.class 的 static final Map<ClassLoader, Map<String, List<String>>> cache = new ConcurrentReferenceHashMap(); 容器中
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 {
// SpringBoot把所有的自動(dòng)裝配的配置類(lèi)都存放在META-INF/spring.factories
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);
}
}
}
spring..factories 概覽
# AutoConfigureCache auto-configuration imports
org.springframework.boot.test.autoconfigure.core.AutoConfigureCache=\
org.springframework.boot.autoconfigure.cache.CacheAutoConfiguration
# AutoConfigureDataCassandra auto-configuration imports
org.springframework.boot.test.autoconfigure.data.cassandra.AutoConfigureDataCassandra=\
org.springframework.boot.autoconfigure.cassandra.CassandraAutoConfiguration,\
org.springframework.boot.autoconfigure.data.cassandra.CassandraDataAutoConfiguration,\
org.springframework.boot.autoconfigure.data.cassandra.CassandraReactiveDataAutoConfiguration,\
org.springframework.boot.autoconfigure.data.cassandra.CassandraReactiveRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.cassandra.CassandraRepositoriesAutoConfiguration
結(jié)論:
Springboot所有自動(dòng)裝配的類(lèi)都是在啟動(dòng)的時(shí)候掃描進(jìn)行加載,通過(guò)加載META-INF/spring.factories文件得到所有類(lèi)的路徑撤蟆,然后通過(guò)@condition條件判斷注解判斷存在該jar包再進(jìn)行自動(dòng)加載奕塑。
具體的層級(jí)和實(shí)現(xiàn)具體看下圖: