SpringBoot詳細(xì)探究講解默認(rèn)組件掃描
參考視頻:https://www.bilibili.com/video/BV1Bq4y1Q7GZ?p=6
通過(guò)視頻的學(xué)習(xí)和自身的理解整理出的筆記。
一、前期準(zhǔn)備
1.1 創(chuàng)建工程
創(chuàng)建springboot項(xiàng)目,springboot版本為2.5.0,引入spring-boot-starter-web依賴,pom文件如下:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.5.0</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<groupId>com.example</groupId>
<artifactId>springboot</artifactId>
<version>0.0.1-SNAPSHOT</version>
<name>springboot</name>
<description>Demo project for Spring Boot</description>
<properties>
<java.version>1.8</java.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>1.2 創(chuàng)建Controller
創(chuàng)建一個(gè)簡(jiǎn)單的Controller用于測(cè)試
@RestController
public class HelloController {
public void helloController() {
System.out.println("創(chuàng)建了");
}
@RequestMapping("hello")
public String hello() {
return "hello";
}
}
二、探究過(guò)程
2.1 探究目標(biāo)
在項(xiàng)目中我們創(chuàng)建了Controller,這個(gè)Controller是如何被spring自動(dòng)加載的呢?為什么Controller必須放在啟動(dòng)類的同級(jí)目錄下呢?
如果我們想要加載不在啟動(dòng)類同級(jí)目錄下的bean對(duì)象,需要在啟動(dòng)類中使用@ComponentScan注解。
目標(biāo):SpringBoot項(xiàng)目中我們沒(méi)有設(shè)置組件掃描的包,為什么它會(huì)默認(rèn)掃描啟動(dòng)類目錄下所有的包。
2.2 探究過(guò)程
2.2.1 回顧容器bean的創(chuàng)建與刷新
在SpringApplication的run()方法中,創(chuàng)建了spring容器context,并通過(guò)refreshContext(context)更新容器加載我們自定義的bean對(duì)象。

我們發(fā)現(xiàn)在執(zhí)行完refreshContext(context)代碼后,自定義的bean對(duì)象(HelloController)就已經(jīng)被創(chuàng)建了,說(shuō)明refreshContext(context)過(guò)程中創(chuàng)建了自定義bean對(duì)象。
下面我們看看究竟是refreshContext(context)中哪些方法創(chuàng)建了自定義bean對(duì)象。
2.2.2 SpringApplication
我接著看refreshContext(context)方法
?? refreshContext()方法

?? refresh()方法

2.2.3 ServletWebServerApplicationContext
再調(diào)用父類的refresh()方法

2.2.4 AbstractApplicationContext
?? refresh()方法

在執(zhí)行完這行代碼后創(chuàng)建了自定義bean的beanDefination對(duì)象。下面來(lái)看看這行代碼。
?? invokeBeanFactoryPostProcessors()方法
根據(jù)這個(gè)名字可以看出來(lái)是調(diào)用了bean工廠的后置處理器。

2.2.5 PostProcessorRegistrationDelegate
?? invokeBeanFactoryPostProcessors()方法
調(diào)用bean工廠的后置處理器,這個(gè)方法很長(zhǎng),最終找到了是這行代碼,調(diào)用BeanDefinition注冊(cè)的后置處理。

?? invokeBeanDefinitionRegistryPostProcessors()方法
拿到后置處理器,調(diào)用后置處理器的BeanDefinition注冊(cè)。

2.2.6 ConfigurationClassPostProcessor
?? postProcessBeanDefinitionRegistry()方法

?? processConfigBeanDefinitions()方法

把啟動(dòng)類的beanDefinition對(duì)象添加到了configCandidates集合中,后面將要用到。

這行代碼執(zhí)行結(jié)束后就有了helloController。
這個(gè)parser是配置類的處理器,通過(guò)傳入很多參數(shù)構(gòu)造了這個(gè)parser處理器。

parser.parse(candidates)中,把啟動(dòng)類對(duì)應(yīng)的beanDefinitionHolder對(duì)象傳進(jìn)去了。
下面看看這個(gè)parse方法。
?? parse()方法

2.2.7 ConfigurationClassParser
?? parse()方法

?? processConfigurationClass()方法

?? doProcessConfigurationClass()方法

if (configClass.getMetadata().isAnnotated(Component.class.getName())) { ... }
判斷啟動(dòng)類上是否加上了@Component注解,這里的if條件成立。
因?yàn)锧SpringBootApplication包含@SpringBootConfiguration,@SpringBootConfiguration包含@Configuration,@Configuration包含@Component,所以加上了@SpringBootApplication注解就相當(dāng)于加上了@Component注解。
?? processMemberClasses()方法
里面有很多處理各類注解的方法
// Process any @PropertySource annotations // Process any @ComponentScan annotations // Process any @Import annotations // Process any @ImportResource annotations // Process individual @Bean methods

后續(xù)將要對(duì)這個(gè)集合進(jìn)行掃描,那么看看它是如何掃描的。
2.2.8 ComponentScanAnnotationParser
?? parse()方法

ClassUtils.getPackageName(declaringClass):獲取啟動(dòng)類所在的包,根據(jù)傳入類的全類名獲取包名。
scanner.doScan(StringUtils.toStringArray(basePackages)):掃描啟動(dòng)類所在的包
2.3 結(jié)論
在容器刷新時(shí)會(huì)調(diào)用BeanFactoryPostProcessor(Bean工廠后置處理器)進(jìn)行處理。其中就有一個(gè)ConfigurationClassPostProcessor(配置類處理器)。在這個(gè)處理器中使用ConfigurationClassParser(配置類解析器)的parse方法去解析處理我們的配置類,其中就有對(duì)ComponentScan注解的解析處理。會(huì)去使用ComponentScanAnnotationParser的parse方法去解析。解析時(shí)如果發(fā)現(xiàn)沒(méi)有配置basePackage,它會(huì)去獲取我們加載了注解的這個(gè)類所在的包,作為我們的basepackage進(jìn)行組件掃描。
到此這篇關(guān)于SpringBoot詳細(xì)探究講解默認(rèn)組件掃描的文章就介紹到這了,更多相關(guān)SpringBoot組件掃描內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java遠(yuǎn)程連接Linux執(zhí)行命令的3種方式完整代碼
在一些Java應(yīng)用程序中需要執(zhí)行一些Linux系統(tǒng)命令,例如服務(wù)器資源查看、文件操作等,這篇文章主要給大家介紹了關(guān)于java遠(yuǎn)程連接Linux執(zhí)行命令的3種方式,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-06-06
Java8進(jìn)行多個(gè)字段分組統(tǒng)計(jì)的實(shí)例代碼
在本篇文章里小編給大家分享的是關(guān)于Java8進(jìn)行多個(gè)字段分組統(tǒng)計(jì)的實(shí)例代碼,需要的朋友們可以學(xué)習(xí)下。2020-05-05
Java獲取當(dāng)?shù)氐娜粘鋈章鋾r(shí)間代碼分享
這篇文章主要介紹了Java獲取當(dāng)?shù)氐娜粘鋈章鋾r(shí)間代碼分享,國(guó)外猿友寫的一個(gè)類,需要的朋友可以參考下2014-06-06
SpringBoot整合FTP實(shí)現(xiàn)文件傳輸?shù)牟襟E
這篇文章主要給大家介紹了SpringBoot整合FTP實(shí)現(xiàn)文件傳輸?shù)牟襟E,文中的流程步驟和代碼示例介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-11-11
SpringBoot整合MOTT動(dòng)態(tài)讀取數(shù)據(jù)庫(kù)連接信息并連接MQTT服務(wù)端
MQTT是一種輕量級(jí)的消息傳輸協(xié)議(Message Queuing Telemetry Transport),旨在實(shí)現(xiàn)設(shè)備之間的低帶寬和高延遲的通信,本文給大家介紹了SpringBoot整合MOTT動(dòng)態(tài)讀取數(shù)據(jù)庫(kù)連接信息并連接MQTT服務(wù)端,需要的朋友可以參考下2024-04-04

