Spring Bean的包掃描的實現(xiàn)方法
我們知道,Spring可以通過包掃描將使用@Component注解定義的Bean定義到容器中。今天就來探究下他實現(xiàn)的原理。
首先,找到@Component注解的處理類
注解的定義,一般都需要配套的對注解的處理才能完成注解所代表的功能。所以我們通過@Component注解的用到的地方,來查找可能的處理邏輯;
我們先進入Spring的項目,在IDEA里面用Ctrl和鼠標左鍵點擊Component注解的名稱,IDEA會顯示出使用到這個類的位置,我們從彈出的列表中找到一個名稱像的類,去看類上面的注釋說明,如圖:

我們點進類中,可以看到第一行就說了這個類是為了從classpath里面找到定義的Bean:

分析具體方法
一般Spring的類都是經(jīng)過設計的,職責清晰。所以一般都是有簡單直接的接口暴露,我們打開類的公開API可以看到有個很直接的方法就叫做掃描,看看注釋說“從指定的包中掃描Bean”,那就是它了。

然后,我們?yōu)榱舜_認,實現(xiàn)確實是通過這個方法,可以啟動程序,打個斷點看看是否經(jīng)過這里(但是這這里,沒有調(diào)用scan()方法,而是更深一層的doScan方法,也確實費解)。
我們進入doScan() 方法看看實現(xiàn):
protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
Assert.notEmpty(basePackages, "At least one base package must be specified");
Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet<>();
// 可以指定多個basePackage,這里就對每個都處理
for (String basePackage : basePackages) {
// 這個方法是真正的查找候選Bean的地方
Set<BeanDefinition> candidates = findCandidateComponents(basePackage);
// 對于每個查找出的候選Bean,進行處理
for (BeanDefinition candidate : candidates) {
// 解析@Scope的元數(shù)據(jù)
ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(candidate);
candidate.setScope(scopeMetadata.getScopeName());
// 為候選的Bean生成一個名稱
String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry);
// 應用后置處理器
if (candidate instanceof AbstractBeanDefinition) {
postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName);
}
//
// 處理一些其它通用的注解的元數(shù)據(jù)
if (candidate instanceof AnnotatedBeanDefinition) {
AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate);
}
// 校驗通過后,注冊到 BeanFactory
if (checkCandidate(beanName, candidate)) {
BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName);
definitionHolder =
AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
beanDefinitions.add(definitionHolder);
registerBeanDefinition(definitionHolder, this.registry);
}
}
}
return beanDefinitions;
}
從方法中我們可以明顯的看到,核心代碼還在findCandidateComponents方法里面,我們進入這個方法后再通過調(diào)試一直找到核心代碼scanCandidateComponents。如下圖,第一處是找到指定包路徑所代表的classpath中的資源對象, 但是這里只是找到了包下面有什么,但是還不知道包下面的類是不是一個候選的Bean(可以看到將DTO類也掃描到了)。如下:

正常思路,拿到了有哪些資源就該進一步去篩選,看看這些資源有哪些是真正的Bean的定義類。
現(xiàn)在我們還不清楚的是,Spring通過什么方式知道一個類是否是真正的Bean的。我們繼續(xù)調(diào)試,到上圖的430行debug進去看看,可以走到org.springframework.core.type.classreading.SimpleMetadataReader這個類的構(gòu)造器中,如下:
SimpleMetadataReader(Resource resource, @Nullable ClassLoader classLoader) throws IOException {
// 通過流讀取資源的內(nèi)容,現(xiàn)在這個資源可以認為是我們的類
InputStream is = new BufferedInputStream(resource.getInputStream());
ClassReader classReader;
try {
// 這個Reader的構(gòu)造器中就將流讀取完畢了
classReader = new ClassReader(is);
}
catch (IllegalArgumentException ex) {
// 通過這個異常的信息,可以推測出,其實這里是通過ASM讀取Class文件的定義了
throw new NestedIOException("ASM ClassReader failed to parse class file - " +
"probably due to a new Java class file version that isn't supported yet: " + resource, ex);
}
finally {
is.close();
}
// 這里根據(jù)命名可以推測是訪問者模式來暴露注解的元數(shù)據(jù)
AnnotationMetadataReadingVisitor visitor = new AnnotationMetadataReadingVisitor(classLoader);
// 這個accpect方法也是訪問者模式中的典型方法,在這里面,是數(shù)據(jù)的解析邏輯
classReader.accept(visitor, ClassReader.SKIP_DEBUG);
this.annotationMetadata = visitor;
// (since AnnotationMetadataReadingVisitor extends ClassMetadataReadingVisitor)
this.classMetadata = visitor;
this.resource = resource;
}
我們在進入classReader.accept方法,這里面可以看到reader對于Class文件的的按字節(jié)解析。

例如,下面讀取的類聲明,類注解都是包掃描需要的類元數(shù)據(jù):

拿到這些元數(shù)據(jù)之后,就按照包掃描的過濾器就過濾出真正需要的類,作為候選的Bean

獲取到元數(shù)據(jù)之后,就可以按部就班對Bean進行注冊、初始化等一系列邏輯啦~
總結(jié)
- 包掃描是通過讀取包對應的類路徑下的
class文件后,對class文件進行解析元數(shù)據(jù)的方式,確定了Bean的定義的; - 本地
IDEA的啟動方式可能和Jar包方式尋找資源的方式略有不同,但是思路是一致的,都是按照第一點查找;
到此這篇關(guān)于Spring Bean的包掃描的實現(xiàn)方法的文章就介紹到這了,更多相關(guān)Spring Bean掃描包內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java JDK與cglib動態(tài)代理有什么區(qū)別
這篇文章主要介紹了Java JDK動態(tài)代理和cglib動態(tài)代理的區(qū)別文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習吧2023-03-03
如何使用spring ResponseEntity處理http響應
這篇文章主要介紹了如何使用spring ResponseEntity處理http響應的操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
SpringBoot實現(xiàn)自定義Redis的連接的流程步驟
Spring Boot 自定義 Redis 主要是指在基于 Spring Boot 的應用程序中,當你需要更深入地控制或擴展對 Redis 數(shù)據(jù)庫的操作,而不是僅僅依賴 Spring Data Redis 的默認配置,本文給大家介紹了SpringBoot實現(xiàn)自定義Redis的連接的流程步驟,需要的朋友可以參考下2024-09-09

