色综合图-色综合图片-色综合图片二区150p-色综合图区-玖玖国产精品视频-玖玖香蕉视频

您的位置:首頁技術文章
文章詳情頁

詳解SpringBoot簡化配置分析總結

瀏覽:4日期:2023-04-18 11:29:39

在SpringBoot啟動類中,該主類被@SpringBootApplication所修飾,跟蹤該注解類,除元注解外,該注解類被如下自定注解修飾。

@SpringBootConfiguration@EnableAutoConfiguration@ComponentScan

讓我們簡單敘述下它們各自的功能:

@ComponentScan:掃描需要被IoC容器管理下需要管理的Bean,默認當前根目錄下的 @EnableAutoConfiguration:裝載所有第三方的Bean @SpringBootConfiguration 作用等同于@Configuration

我們來看下@SpringBootConfiguration

@Target({ElementType.TYPE})@Retention(RetentionPolicy.RUNTIME)@Documented@Configurationpublic @interface SpringBootConfiguration { @AliasFor( annotation = Configuration.class ) boolean proxyBeanMethods() default true;}

可以看到該注解類內包含與@Configuration,其作用與@Configuration并無太大區別,只是多了層屬性嵌套。

故: @SpringBootConfiguration + @ComponentScan

將根目錄下所有被**@Controller、@Service、@Repository、@Component**等所修飾的類交給IoC容器管理。

那么重點來了,@EnableAutoConfiguration是如何裝載第三方Bean的呢?讓我們跟蹤下它的源碼。

首先我們可以看到該類被如下注解修飾:

@AutoConfigurationPackage@Import({AutoConfigurationImportSelector.class})

我們先關注下AutoConfigurationImportSelector這個組件。

// 批量導入第三方的一些Bean@Import({AutoConfigurationImportSelector.class})

其中該組件的selectImports(AnnotationMetadata annotationMetadata)方法,我們先簡述下它的作用:掃描所有需要被管理的第三方Bean并交給IoC容器進行管理。然后我們接著往下追蹤。

public String[] selectImports(AnnotationMetadata annotationMetadata) { if (!this.isEnabled(annotationMetadata)) { return NO_IMPORTS; } else { // 讓我們跟蹤到這個方法 AutoConfigurationImportSelector.AutoConfigurationEntry autoConfigurationEntry = this.getAutoConfigurationEntry(annotationMetadata); return StringUtils.toStringArray(autoConfigurationEntry.getConfigurations()); }}

protected AutoConfigurationImportSelector.AutoConfigurationEntry getAutoConfigurationEntry(AnnotationMetadata annotationMetadata) { if (!this.isEnabled(annotationMetadata)) { return EMPTY_ENTRY; } else { AnnotationAttributes attributes = this.getAttributes(annotationMetadata); // 獲取所有AutoConfiguration的配置類 List<String> configurations = this.getCandidateConfigurations(annotationMetadata, attributes); // 下面就是對AutoConfiguration的去重、排除和過濾等操作 configurations = this.removeDuplicates(configurations); Set<String> exclusions = this.getExclusions(annotationMetadata, attributes); this.checkExcludedClasses(configurations, exclusions); configurations.removeAll(exclusions); configurations = this.getConfigurationClassFilter().filter(configurations); // 我們繼續追蹤這里 this.fireAutoConfigurationImportEvents(configurations, exclusions); return new AutoConfigurationImportSelector.AutoConfigurationEntry(configurations, exclusions); }}

private void fireAutoConfigurationImportEvents(List<String> configurations, Set<String> exclusions) { List<AutoConfigurationImportListener> listeners = this.getAutoConfigurationImportListeners(); if (!listeners.isEmpty()) { // 加了層包裝 AutoConfigurationImportEvent event = new AutoConfigurationImportEvent(this, configurations, exclusions); Iterator var5 = listeners.iterator(); while(var5.hasNext()) { AutoConfigurationImportListener listener = (AutoConfigurationImportListener)var5.next(); this.invokeAwareMethods(listener); // 向ConditionEvaluationReport中導入所有AutoConfiguration listener.onAutoConfigurationImportEvent(event); } }}

可以猜想IoC容器在啟動時會將這里的AutoConfiguration中的每個Bean都注入到容器中。這里的源碼我們先跟蹤到這里,大致了解了下該方法的作用。

那么SpringBoot又是如何取感知第三方的Bean文件呢?

SpringBoot和第三方Bean之間存在一定的規定。即通過對于相應依賴的Jar包中可能存在一個spring.factories文件,在該文件中就記錄了需要被IoC容器管理的Bean文件路徑,SpringBoot通過該文件確定需要IoC管理的Bean文件位置。對于spring-boot-autoconfiguration的spring.factories文件中,記錄著大量xxxAutoConfiguration的類文件位置,這些類都被@Configuration注解標識,即這些配置類會配置多個Bean從而解決spring.factories可能產生的臃腫問題。

Tomcat的加載時機

對于SpringBoot來說它特點不僅是簡化配置,還有內嵌容器等特點。那么就有必要探討Tomcat容器的加載時機。在spring-boot-autoconfiguration的spring.factories文件中存在ServletWebServerFactoryAutoConfiguration配置類的路徑,該類會在項目啟動時將默認的Tomcat容器已@Bean的方式加載入IoC容器內部。

SpringBoot是如何集中配置呢?

談論這個問題前我們不妨先按照之前yml或properties的文件配置下

server: port: 8080

通過IDE,跟蹤到port所配置的成員變量所在類,發現該類被@ConfigurationProperties所修飾,該注解就是將yml或properties中配置按照對應前綴注入到指定類的成員變量。該注解具體實現感興趣的小伙伴們可以去如下鏈接學習。 @ConfigurationProperties實現原理與實戰

@ConfigurationProperties(prefix = 'server', ignoreUnknownFields = true)public class ServerProperties { private Integer port;*******}

下面兩個代碼和前述作用大致相同

environment.getProperty('xxx');@Value('${xxx}')

我們在使用SpringBoot時只需要做哪些事情?

通常我們再使用SpringBoot時只需要在Maven中引入類似如下的starter依賴。

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

最多再需要配置一些類似mybatis這類框架的一些屬性參數。而這些starter按照我們之前的邏輯其內部應該存有spring.factories文件,我們先去對應jar包查找下。

如果有些starter的jar包沒有找到我們想要的spring.factories文件。我們可以去spring-boot-test-autoconfiguretion中的spring.factories查看下,SpringBoot內部其實已經定義好相當一定數量的AutoConfiguration。

詳解SpringBoot簡化配置分析總結

果然該jar包內確實存在spring.factories文件,代碼如下。

org.springframework.data.repository.core.support.RepositoryFactorySupport=org.springframework.data.redis.repository.support.RedisRepositoryFactory

這意味著我們已經簡單地了解了SpringBoot如何簡化配置,那么我們也應該可以自己來實現一個starter依賴交給SpringBoot來使用,只要在對應Jar包中添加spring.factories文件,在其中添加如下代碼。

org.springframework.boot.autoconfigure.EnableAutoConfiguration=xxxAutoConfiguration

大家若有時間還請實現下自己的starter依賴,對加深這部分理解還是很有幫助的。感興趣的小伙伴可以看下我做的一個簡單的實現。 [自定義starter實現]

最后我們在說下最后@SpringBootApplication中@AutoConfigurationPackage這個注解類,發現其中導入了Registrar組件。

@Import({Registrar.class})

讓我們重點關注registerBeanDefinitions這個方法,該方法最終會來到DefaultListableBeanFactory中registerBeanDefinition(String beanName, BeanDefinition beanDefinition)方法,將AutoConfigurationPackages.class注冊到IoC容器中,然后將主配置類所在包下所有組件導入到SpringIoC容器中

public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) { // 里面就這一個方法我們跟蹤下 AutoConfigurationPackages.register(registry, (String[])(new AutoConfigurationPackages.PackageImports(metadata)).getPackageNames().toArray(new String[0]));}

public static void register(BeanDefinitionRegistry registry, String... packageNames) { // 判斷beanDefinitionMap是否存在AutoConfigurationPackages if (registry.containsBeanDefinition(BEAN)) { BeanDefinition beanDefinition = registry.getBeanDefinition(BEAN); ConstructorArgumentValues constructorArguments = beanDefinition.getConstructorArgumentValues(); constructorArguments.addIndexedArgumentValue(0, addBasePackages(constructorArguments, packageNames)); } else { GenericBeanDefinition beanDefinition = new GenericBeanDefinition(); beanDefinition.setBeanClass(AutoConfigurationPackages.BasePackages.class); beanDefinition.getConstructorArgumentValues().addIndexedArgumentValue(0, packageNames); beanDefinition.setRole(2); // 將設置好的AutoConfigurationPackages注冊到beanDefinitionMap(是不是很熟悉這一步) registry.registerBeanDefinition(BEAN, beanDefinition); }}

怎么樣,在為我們簡化了配置的同時,SpringBoot居然幫我們做了如此多的事情,而我們只需要簡單地集中配置其中一部分的屬性。關于SpirngBoot我們就探討到這里,這些內容是閱讀一些文章,觀看部分講解和源碼的總結,若有錯誤還請接納與指教。這是本人的第一篇文章,最后感謝各位的閱讀。

到此這篇關于詳解SpringBoot簡化配置分析總結的文章就介紹到這了,更多相關SpringBoot 簡化配置內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Spring
相關文章:
主站蜘蛛池模板: 午夜性色福利视频在线视频 | 国产在线观看成人免费视频 | 黄色网址视频在线观看 | 伊人狼人影院 | 久久99热成人精品国产 | 国产精品亚洲片夜色在线 | 我要看a级毛片 | 亚洲天堂网在线播放 | 国产成人综合怡春院精品 | 久久亚洲精品一区成人 | 91久久香蕉国产线看观看软件 | 91成人免费在线视频 | 免费观看欧美一区二区三区 | 国产不卡一区二区三区免费视 | 免费黄色一级网站 | 禁止18周岁进入免费网站观看 | 国产成人在线免费视频 | 一个人的视频日本免费 | 国产高清亚洲 | 国产成人精品久久 | 国产真实女人一级毛片 | 毛片在线全部免费观看 | 国产三级免费观看 | 97视频免费在线观看 | 九九九热在线精品免费全部 | 天堂一区二区三区在线观看 | 最近中文字幕在线 | 中文 | 中文字幕人成不卡一区 | 国产三级做爰高清视频a | 一级在线毛片 | 久久成年片色大黄全免费网站 | 亚洲精品午夜一区二区在线观看 | 亚洲成a人片毛片在线 | 亚洲精品中文一区不卡 | 欧美二级在线观看免费 | 久久亚洲网 | 九九re6精品视频在线观看 | 一区二区国产在线观看 | 一区二区三区在线看 | 久草精彩视频 | 成人三级精品视频在线观看 |