您好,登錄后才能下訂單哦!
這篇文章給大家介紹利用Spring Bean怎么實現(xiàn)一個包掃描功能,內容非常詳細,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。
首先,找到@Component注解的處理類
注解的定義,一般都需要配套的對注解的處理才能完成注解所代表的功能。所以我們通過@Component
注解的用到的地方,來查找可能的處理邏輯;
我們先進入Spring
的項目,在IDEA
里面用Ctrl
和鼠標左鍵點擊Component
注解的名稱,IDEA
會顯示出使用到這個類的位置,我們從彈出的列表中找到一個名稱像的類,去看類上面的注釋說明,如圖:
我們點進類中,可以看到第一行就說了這個類是為了從classpath
里面找到定義的Bean
:
一般Spring
的類都是經過設計的,職責清晰。所以一般都是有簡單直接的接口暴露,我們打開類的公開API
可以看到有個很直接的方法就叫做掃描,看看注釋說“從指定的包中掃描Bean”,那就是它了。
然后,我們?yōu)榱舜_認,實現(xiàn)確實是通過這個方法,可以啟動程序,打個斷點看看是否經過這里(但是這這里,沒有調用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
方法里面,我們進入這個方法后再通過調試一直找到核心代碼scanCandidateComponents
。如下圖,第一處是找到指定包路徑所代表的classpath
中的資源對象, 但是這里只是找到了包下面有什么,但是還不知道包下面的類是不是一個候選的Bean
(可以看到將DTO
類也掃描到了)。如下:
正常思路,拿到了有哪些資源就該進一步去篩選,看看這些資源有哪些是真正的Bean
的定義類。
現(xiàn)在我們還不清楚的是,Spring
通過什么方式知道一個類是否是真正的Bean
的。我們繼續(xù)調試,到上圖的430行debug
進去看看,可以走到org.springframework.core.type.classreading.SimpleMetadataReader
這個類的構造器中,如下:
SimpleMetadataReader(Resource resource, @Nullable ClassLoader classLoader) throws IOException { // 通過流讀取資源的內容,現(xiàn)在這個資源可以認為是我們的類 InputStream is = new BufferedInputStream(resource.getInputStream()); ClassReader classReader; try { // 這個Reader的構造器中就將流讀取完畢了 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
關于利用Spring Bean怎么實現(xiàn)一個包掃描功能就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經查實,將立刻刪除涉嫌侵權內容。