springboot自動裝配原理初識
運(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>
啟動器:spring-boot-starter-xxx:springboot的場景啟動器
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方法啟動的springboot應(yīng)用。
@SpringBootApplication //標(biāo)注這是一個(gè)主程序類,說明這是一個(gè)springboot應(yīng)用 public class Springboot01HelloworldApplication { public static void main(String[] args) { //這里啟動了一個(gè)服務(wù),而不是執(zhí)行了一個(gè)方法。 SpringApplication.run(Springboot01HelloworldApplication.class, args); } }
點(diǎn)@SpringBootApplication繼續(xù)研究,會發(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自動掃描包
這個(gè)我們在spring配置文件中見到過,它用來自動掃描并加載符合條件的組件或者bean,并將bean加載到IOC容器中。
2.@SpringBootConfiguration: springboot的配置類
標(biāo)注在某個(gè)類上,說明這個(gè)類是springboot的配置類,在這里它就說明SpringBootApplication這個(gè)類是springboot的配置類。
我們繼續(xù)點(diǎn)@SpringBootConfiguration進(jìn)去查看,會發(fā)現(xiàn) @Configuration這個(gè)注解
2.1 @Configuration:配置類,用來配置spring的xml文件
我們繼續(xù)點(diǎn)@Configuration進(jìn)去查看,會發(fā)現(xiàn) @Component這個(gè)注解。
2.2 @Component:組件,說明啟動類本身也是一個(gè)組件,負(fù)責(zé)啟動應(yīng)用。
至此,@SpringBootConfiguration這條線,我們研究完了。
3.@EnableAutoConfiguration:開啟自動裝配,通過@EnableAutoConfiguration來幫我們自動配置之前我們需要配置的東西。
我們繼續(xù)點(diǎn)@EnableAutoConfiguration進(jìn)去查看,會發(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自動裝配包
繼續(xù)點(diǎn)進(jìn)去查看,出現(xiàn)@Import({Registrar.class})這個(gè)注解
3.1.1 @Import({Registrar.class}): spring底層注解,給容器導(dǎo)入一個(gè)組件
Registrar.class: 將主啟動類所在包及所在包下面的所有子包里面所有的組件都掃描到Spring容器。
至此,@AutoConfigurationPackage這條線我們也研究完了。
3.2 @Import({AutoConfigurationImportSelector.class}): 給容器導(dǎo)入組件
AutoConfigurationImportSelector.class:自動裝配導(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()方法,返回一開始我們看到的啟動自動配置文件的注解類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
我們在上面的自動配置類隨便找一個(gè)打開看看,比如 :WebMvcAutoConfiguration
都是大家熟悉的配置,所以,自動配置真正實(shí)現(xiàn)是從classpath中搜尋所有的META-INF/spring.factories配置文件 ,并將其中對應(yīng)的 org.springframework.boot.autoconfigure. 包下的配置項(xiàng),通過反射實(shí)例化為對應(yīng)標(biāo)注了 @Configuration的JavaConfig形式的IOC容器配置類 , 然后將這些都匯總成為一個(gè)實(shí)例并加載到IOC容器中。
總結(jié)
- SpringBoot在啟動的時(shí)候從類路徑下的META-INF/spring.factories中獲取EnableAutoConfiguration指定的值
- 將這些值作為自動配置類導(dǎo)入容器,自動配置類就生效,幫我們進(jìn)行自動配置工作;
- 整個(gè)J2EE的整體解決方案和自動配置都在springboot-autoconfigure的jar包中;
- 它會給容器中導(dǎo)入非常多的自動配置類 (xxxAutoConfiguration), 就是給容器中導(dǎo)入這個(gè)場景需要的所有組件 , 并配置好這些組件 ;
- 有了自動配置類 , 免去了我們手動編寫配置注入功能組件等的工作;
主啟動類
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對象
SpringApplication主要做的事情:
- 推斷應(yīng)用的類型是普通的項(xiàng)目還是Web項(xiàng)目
- 查找并加載所有可用初始化器 , 設(shè)置到initializers屬性中
- 找出所有的應(yīng)用程序監(jiān)聽器,設(shè)置到listeners屬性中
- 推斷并設(shè)置main方法的定義類,找到運(yùn)行的主類
以上就是springboot自動裝配原理初識的詳細(xì)內(nèi)容,更多關(guān)于springboot自動裝配原理的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
java正則表達(dá)式簡單使用和網(wǎng)頁爬蟲的制作代碼
java正則表達(dá)式簡單使用和網(wǎng)頁爬蟲的制作代碼,需要的朋友可以參考一下2013-05-05解讀HttpServletRequestWrapper處理request數(shù)據(jù)流多次讀取問題
在Java Web開發(fā)中,獲取HTTP請求參數(shù)是常見需求,本文詳細(xì)討論了通過POST方式獲取參數(shù)的兩種主要方法:使用request.getParameter()適用于application/x-www-form-urlencoded和multipart/form-data內(nèi)容類型;而對于application/json類型的數(shù)據(jù)2024-10-10詳解SpringBoot中的統(tǒng)一結(jié)果返回與統(tǒng)一異常處理
這篇文章主要將通過詳細(xì)的討論和實(shí)例演示來幫助你更好地理解和應(yīng)用Spring Boot中的統(tǒng)一結(jié)果返回和統(tǒng)一異常處理,感興趣的小伙伴可以了解下2024-03-03Spring Cloud中Eureka開啟密碼認(rèn)證的實(shí)例
這篇文章主要介紹了Spring Cloud中Eureka開啟密碼認(rèn)證的實(shí)例,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-05-05Jaxb2實(shí)現(xiàn)JavaBean與xml互轉(zhuǎn)的方法詳解
這篇文章主要介紹了Jaxb2實(shí)現(xiàn)JavaBean與xml互轉(zhuǎn)的方法,簡單介紹了JAXB的概念、功能及實(shí)現(xiàn)JavaBean與xml互轉(zhuǎn)的具體操作技巧,需要的朋友可以參考下2017-04-04Java 連接Access數(shù)據(jù)庫的兩種方式
這篇文章主要介紹了Java 連接Access數(shù)據(jù)庫的兩種方式,本文著重講解使用JDBC連接操作Access數(shù)據(jù)庫,需要的朋友可以參考下2015-06-06JAVA異常信息Exception?e及e的相關(guān)方法解讀
這篇文章主要介紹了JAVA異常信息Exception?e及e的相關(guān)方法解讀,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-06-06