日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當前位置:首頁 > 科技  > 軟件

一篇學會SpringBoot自動裝配

來源: 責編: 時間:2023-11-01 09:17:46 301觀看
導讀一.何為自動裝配自動裝配是 SpringBoot 的核心功能,主要是讓開發者盡可能少的關注一些基礎化的 Bean 的配置,實際上完成的工作是如何自動將 Bean 裝載到 Ioc 容器中。在 SpringBoot 中如果想要引入一個新的模塊,例如項目

一.何為自動裝配

自動裝配是 SpringBoot 的核心功能,主要是讓開發者盡可能少的關注一些基礎化的 Bean 的配置,實際上完成的工作是如何自動將 Bean 裝載到 Ioc 容器中。LF328資訊網——每日最新資訊28at.com

在 SpringBoot 中如果想要引入一個新的模塊,例如項目中想使用 Redis 緩存,只需要做以下幾步即可。LF328資訊網——每日最新資訊28at.com

1、在 pom.xml 文件中引入 spring-boot-starter-data-redis 相關的 jar 包LF328資訊網——每日最新資訊28at.com

<dependency>  <groupId>org.springframework.boot</groupId>  <artifactId>spring-boot-starter-data-redis</artifactId></dependency>

2、在 application.properties 文件中加入 Redis 相關的配置LF328資訊網——每日最新資訊28at.com

spring.redis.host=127.0.0.1spring.redis.port=6379

3、在代碼中引用 Redis 緩存的操作類LF328資訊網——每日最新資訊28at.com

@Autowiredprivate RedisTemplate<String,String>redisTemplate;

為什么 RedisTemplate 可以被直接注入,它是什么時候加入到 Ioc 容器中的,這都是自動裝配的功勞,我們一起來看一下。LF328資訊網——每日最新資訊28at.com

二.自動裝配過程

1、@EnableAutoConfiguration分析

SpringBoot 項目啟動類上有 @SpringBootApplication 這樣一個注解,它繼承了 @EnableAutoConfiguration,主要作用是幫助 Springboot 應用把所有符合條件的配置類都加載到當前 SpringBoot 創建并使用的 Ioc 容器中。LF328資訊網——每日最新資訊28at.com

這個注解主要由兩部分組成LF328資訊網——每日最新資訊28at.com

  • @AutoConfigurationPackage,指定 SpringBoot 掃描的包范圍,這個范圍下使用 @Service 、 @Component 等注解的 Bean 加入 Ioc 容器,默認值是啟動類所在的包路徑,默認指定啟動類路徑下的類加載到 Ioc 容器。
  • **@Import(AutoConfigurationImportSelector.class)**,將導入第三方提供的 Bean 配置類加載加載到 Ioc 容器。
@AutoConfigurationPackage@Import(AutoConfigurationImportSelector.class)public @interface EnableAutoConfiguration {  ...}

1.1 @AutoConfigurationPackage

@AutoConfigurationPackage 指定 SpringBoot 掃描的包范圍,主要邏輯在 AutoConfigurationPackages#register 方法中。LF328資訊網——每日最新資訊28at.com

該方法有兩個參數 registry 和 packageNames,在斷點中發現 registry 實際上就是 DefaultListableBeanFactory 實例,packageNames 的值默認是啟動類包所在的路徑,在這里將 @AutoConfigurationPackage 指定的包路徑添加到 DefaultListableBeanFactory,在后續Ioc容器掃描時將其加載進去。LF328資訊網——每日最新資訊28at.com

圖片圖片LF328資訊網——每日最新資訊28at.com

1.2 AutoConfigurationImportSelector.class

AutoConfigurationImportSelector 主要是實現 importSelector 方法來實現基于動態 Bean 的加載功能,我們定位到 importSelector 方法看一下里面的邏輯。LF328資訊網——每日最新資訊28at.com

@Overridepublic String[] selectImports(AnnotationMetadata annotationMetadata) {  if (!isEnabled(annotationMetadata)) {    return NO_IMPORTS;  }  //1、從配置文件spring-autoconfigure-metadata.properties中加載自動裝配候選規則  AutoConfigurationMetadata autoConfigurationMetadata = AutoConfigurationMetadataLoader    .loadMetadata(this.beanClassLoader);  //2、獲取@SpringBootApplication上配置的屬性值 AnnotationAttributes attributes = getAttributes(annotationMetadata);  //3、使用SpringFactoriesLoader 加載classpath路徑下META-INF/spring.factories中  //通過key=org.springframework.boot.autoconfigure.EnableAutoConfiguration獲取候選類 List<String> configurations = getCandidateConfigurations(annotationMetadata,attributes);  //4、去除重復值  configurations = removeDuplicates(configurations);  //5、獲取exclude屬性值,將exclude中的值排除掉  Set<String> exclusions = getExclusions(annotationMetadata, attributes);  checkExcludedClasses(configurations, exclusions);  configurations.removeAll(exclusions);  //6、檢查候選配置類上的注解@ConditionalOnClass,如果要求的類不存在,則這個候選類會被過濾不被加載  configurations = filter(configurations, autoConfigurationMetadata);  fireAutoConfigurationImportEvents(configurations, exclusions);  return StringUtils.toStringArray(configurations);}

第一步和第三步邏輯中涉及到兩個非常重要的文件 spring-autoconfigure-metadata.properties、spring.factoriesLF328資訊網——每日最新資訊28at.com

  • spring.factories 主要記錄了待自動裝配的候選類,從下圖斷點中可以看到有 109 個。

圖片圖片LF328資訊網——每日最新資訊28at.com

  • spring-autoconfigure-metadata.properties 中配置了系列 ConditionalOnClass 類和配置類之間的依賴,通過這個文件中配置的規則,來判斷 spring.factories 哪些類真的需要加載。為什么會這樣做,是因為很多的 @Configuration 其實是依托于其他的框架來加載的,如果當前的 classpath 環境下沒有相關聯的依賴,則意味著這些類沒必要進行加載,斷點的最后發現真正需要加載的只有 31 個。

圖片LF328資訊網——每日最新資訊28at.com

  • 加載 spring.factories 文件,借助了 Spring 框架提供的一個工具類 SpringFactoriesLoader,它的實現和 java 中的 SPI 機制原理是一樣的,它相對于 SPI 的改進點在于不會一次性加載所有的類,而是根據 key 進行加載。

SPI ,全稱為 Service Provider Interface,是一種服務發現機制。它通過在 ClassPath 路徑下的 META-INF/services 文件夾查找文件,自動加載文件里所定義的類。這一機制為很多框架擴展提供了可能,比如在 Dubbo、JDBC 中都使用到了 SPI 機制。LF328資訊網——每日最新資訊28at.com

圖片圖片LF328資訊網——每日最新資訊28at.com

2、@Import分析

在 @EnableAutoConfiguration 分析中,兩種加載 Bean 到 Ioc 容器的方式,他們都是通過 @import 引入,這里我們來分析一下 @import 是在哪里進行加載的。LF328資訊網——每日最新資訊28at.com

2.1 @import3種使用方式

  • 引入普通類,直接在 @Import 中引入即可。
@Import(PersonConfig.class)@Configurationpublic class PersonConfiguration { }
  • 引入 ImportSelector 實現類(AutoConfigurationImportSelector 就是實現了這個接口),這個接口需要實現的方法是 selectImports(),它返回的是一個字符串數組,代表的是類名集合,這些類將會被加載到Ioc容器中。
@Import(TestImportSelector.class)@Configurationpublic class ImportTestConfig {}
public class TestImportSelector implements ImportSelector {  @Override  public String[] selectImports(AnnotationMetadata importingClassMetadata) {    return new String[]{"com.example.service.TestService"};  }}
  • 引用 ImportBeanDefinitionRegistrar 實現類(AutoConfigurationPackages.Registrar 就是實現類這個接口),這個接口需要實現的方法是 registerBeanDefinitions(),它有兩個入參,第一個參數 AnnotationMetadata代表當前類的注解信息;第二個參數 registry 代表的是 DefaultListableBeanFactory 實例,因為參數 DefaultListableBeanFactory 代表的是 Ioc 容器,如果想注入 Bean,可以直接對該類進行操作。
@Import(TestImportBeanDefinitorSelector.class)@Configurationpublic class ImportBeanDefinitionTestConfig {}
public class TestImportBeanDefinitorSelector implements ImportBeanDefinitionRegistrar {  @Override  public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata,                                      BeanDefinitionRegistry registry) {    AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder     .rootBeanDefinition(Person.class)     .getBeanDefinition();    registry.registerBeanDefinition("person", beanDefinition);  }}

2.2 @Import實現原理

  • @Import 主要由 ConfigurationClassPostProcessor 后置處理器加載實現,ConfigurationClassPostProcessor 則實現了 BeanDefinitionRegistryPostProcessor 接口,在 invokeBeanFactoryPostProcessors() 方法中進行處理,該方法實例化并調用所有的 BeanFactory 后置處理器。
@Overridepublic void refresh() throws BeansException, IllegalStateException {  //....省略n行代碼  //1.beanFactory后置處理邏輯,在這個方法里加載ConfigurationClassPostProcessor  invokeBeanFactoryPostProcessors(beanFactory);  //2.注冊bean后置處理邏輯  registerBeanPostProcessors(beanFactory);  //...省略n行代碼  //3.實例化非懶加載的bean,并加入到Ioc容器中  finishBeanFactoryInitialization(beanFactory);  //....省略n行代碼}
  • ConfigurationClassPostProcessor 實現了方法 postProcessBeanDefinitionRegistry(),在這個方法中跟蹤代碼到 ConfigurationClassParser.parse(),所有配置類的解析邏輯都在 parse() 方法中進行處理。
  • 在 ConfigurationClassParser.parseparse() 繼續往下跟蹤會到 doProcessConfigurationClass() 方法,在該方法中會有一些常用配置注解的解析,例如 @Component、@ComponentScan、@Bean、@Configuration、@Import 等。
@Nullableprotected final SourceClass doProcessConfigurationClass(ConfigurationClass configClass, SourceClass sourceClass)   throws IOException {  //...省略n行代碼  //加載@Import注解,遞歸解析,獲取導入的配置類  processImports(configClass, sourceClass, getImports(sourceClass), true);  //...省略n行代碼}
  • processImports() 中主要實現類了 @Import 接口的 3 種不同的加載方式
private void processImports(ConfigurationClass configClass,                             SourceClass currentSourceClass,                            Collection<SourceClass> importCandidates,                             boolean checkForCircularImports) {  //...省略n行代碼  if (candidate.isAssignable(ImportSelector.class)) {   //1.實現了ImportSelector接口的類在@Import中引用邏輯   Class<?> candidateClass = candidate.loadClass();   ImportSelector selector = BeanUtils.instantiateClass(      candidateClass,ImportSelector.class);   ParserStrategyUtils.invokeAwareMethods(     selector, this.environment, this.resourceLoader, this.registry);   if (this.deferredImportSelectors != null &&       selector instanceof DeferredImportSelector) {     this.deferredImportSelectors.add(      new DeferredImportSelectorHolder(configClass, (DeferredImportSelector) selector));   } else {     String[] importClassNames = selector.selectImports(currentSourceClass.getMetadata());    Collection<SourceClass> importSourceClasses = asSourceClasses(importClassNames);    processImports(configClass, currentSourceClass, importSourceClasses, false);   }  } else if (candidate.isAssignable(ImportBeanDefinitionRegistrar.class)) {   //2.實現了ImportBeanDefinitionRegistrar接口的類在@Import中引用邏輯   Class<?> candidateClass = candidate.loadClass();   ImportBeanDefinitionRegistrar registrar =BeanUtils.instantiateClass(      candidateClass, ImportBeanDefinitionRegistrar.class);  ParserStrategyUtils.invokeAwareMethods(    registrar, this.environment, this.resourceLoader, this.registry);  configClass.addImportBeanDefinitionRegistrar(      registrar, currentSourceClass.getMetadata()); } else {    //3.普通類直接在@Import中引用邏輯   this.importStack.registerImport(      currentSourceClass.getMetadata(), candidate.getMetadata().getClassName());  processConfigurationClass(candidate.asConfigClass(configClass)); }  //...省略n行代碼}

總結一下就是如下的方法鏈調用LF328資訊網——每日最新資訊28at.com

refresh()=>invokeBeanFactoryPostProcessors()=>postProcessBeanDefinitionRegistry()=>parse()=>doProcessConfigurationClass()=>processImports()

3、啟動類何時加入到 Ioc 容器

前面我們分析了自動裝配的主要邏輯,那么 SpringBoot 啟動類又是如何加入到Ioc容器中的呢?LF328資訊網——每日最新資訊28at.com

3.1 prepareContext() 中的 load() 方法

  • 從 SpringBoot 啟動類的 run() 方法開始,跟蹤代碼到 SpringApplication.run() 方法,這里是 SpringBoot 啟動的核心邏輯。
  • 在 SpringApplication.run() 方法中有一個 prepareContext() 方法,進入這個方法里面,會發現有一個 load() 方法,這里就是加載啟動類的地方,它會將啟動類注入到 Ioc 容器中。
private void prepareContext(ConfigurableApplicationContext context,   ConfigurableEnvironment environment, SpringApplicationRunListeners listeners,   ApplicationArguments applicationArguments, Banner printedBanner) {  //...省略n行代碼  //加載啟動類,將啟動類注入到Ioc容器中  load(context, sources.toArray(new Object[0]));  //...省略n行代碼}
  • 在斷點中可以看到,SpringBoot 啟動類注入到了 annotatedReader 中(AnnotatedBeanDefinitionReader 基于注解的 beanDefinition 解析器),在這里將啟動類加入到了 Ioc 容器。

圖片圖片LF328資訊網——每日最新資訊28at.com

總結一下就是如下的方法鏈調用LF328資訊網——每日最新資訊28at.com

run()=>prepareContext()()=>load()=>parse()=>register()

4、自動裝配整體流程

基于以上3塊的分析我們可以得到如下一個關于自動裝配的流程圖LF328資訊網——每日最新資訊28at.com

圖片圖片LF328資訊網——每日最新資訊28at.com

三.總結

學習源碼的過程中如果不了解源碼的整體思路,直接看代碼會迷失在源碼的海洋中。要了解代碼的整體脈絡,以總-分-總的方式去學習,學會舍棄部分無關的代碼,才能高效的閱讀和學習源碼,從中汲取到代碼的精華所在,提升自己的編程能力。LF328資訊網——每日最新資訊28at.com

參考資料:

  • 知乎:@Import使用及原理詳解

本文鏈接:http://www.www897cc.com/showinfo-26-16254-0.html一篇學會SpringBoot自動裝配

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: 大家的平原,阿里云的播種機

下一篇: 深入解析冪等性在Python開發中的應用

標簽:
  • 熱門焦點
  • 小米官宣:2023年上半年出貨量中國第一!

    今日早間,小米電視官方微博帶來消息,稱2023年小米電視上半年出貨量達到了中國第一,同時還表示小米電視的巨屏風暴即將開始?!肮家粋€好消息2023年#小米電視上半年出貨量中國
  • 印度登月最關鍵一步!月船三號今晚進入環月軌道

    8月5日消息,據印度官方消息,月船三號將于北京時間今晚21時30分左右開始近月制動進入環月軌道。這是該探測器能夠成功的最關鍵步驟之一,如果成功將開始圍
  • 分享六款相見恨晚的PPT模版網站, 祝你做出精美的PPT!

    1、OfficePLUSOfficePLUS網站旨在為全球Office用戶提供豐富的高品質原創PPT模板、實用文檔、數據圖表及個性化定制服務。優點:OfficePLUS是微軟官方網站,囊括PPT模板、Word模
  • 每天一道面試題-CPU偽共享

    前言:了不起:又到了每天一到面試題的時候了!學弟,最近學習的怎么樣啊 了不起學弟:最近學習的還不錯,每天都在學習,每天都在進步! 了不起:那你最近學習的什么呢? 了不起學弟:最近在學習C
  • 年輕人的“職場羞恥感”,無處不在

    作者:馮曉亭 陶 淘 李 欣 張 琳 馬舒葉來源:燃次元&ldquo;人在職場,應該選擇什么樣的著裝?&rdquo;近日,在網絡上,一個與著裝相關的帖子引發關注,在該帖子里,一位在高級寫字樓亞洲金
  • 三星獲批量產iPhone 15全系屏幕:蘋果史上最驚艷直屏

    按照慣例,蘋果將繼續在今年9月舉辦一年一度的秋季新品發布會,有傳言稱發布會將于9月12日舉行,屆時全新的iPhone 15系列將正式與大家見面,不出意外的話
  • 2299元起!iQOO Pad明晚首銷:性能最強天璣平板

    5月23日,iQOO如期舉行了新品發布會,除了首發安卓最強旗艦處理器的iQOO Neo8系列新機外,還在發布會上推出了旗下首款平板電腦——iQOO Pad,其最大的賣點
  • 電博會與軟博會實現"線下+云端"的雙線融合

    在本次“電博會”與“軟博會”雙展會利好條件的加持下,既可以發揮展會拉動人流、信息流、資金流實現快速交互流動的作用,繼而推動區域經濟良性發展;又可以聚
  • 親歷馬斯克血洗Twitter,硅谷的苦日子在后頭

    文/劉哲銘  編輯/李薇  馬斯克再次揮下裁員大刀?! ∶绹鴷r間11月14日,Twitter約4400名外包員工遭解雇,此次被解雇的員工的主要工作為內容審核等。此前,T
Top 主站蜘蛛池模板: 涿州市| 休宁县| 沙田区| 曲水县| 枝江市| 资源县| 北京市| 平顶山市| 当涂县| 平凉市| 木兰县| 苗栗市| 龙胜| 南开区| 赤峰市| 嘉禾县| 巩义市| 麟游县| 龙南县| 南城县| 盐津县| 无锡市| 广州市| 冷水江市| 永年县| 耿马| 泗水县| 凌海市| 张北县| 米易县| 德昌县| 上思县| 甘孜| 南京市| 蒲江县| 福泉市| 长兴县| 梁山县| 海伦市| 安化县| 封开县|