成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

你知道什么是 @Component 注解的派生性嗎?

開發 前端
Spring Framework? 每個版本的具體實現會有差異,阿粉使用的版本是 5.3.24 ,所以如果小伙伴看到自己的代碼追蹤的效果跟阿粉的不一樣也不會奇怪,可能是因為版本不一樣而已,不過本質上都是一樣的。

對于 @Component? 注解在日常的工作中相信很多小伙伴都會使用到,作為一種 Spring? 容器托管的通用模式組件,任何被 @Component? 注解標注的組件都會被 Spring 容器掃描。

那么有的小伙伴就要問了,很多時候我們并沒有直接寫 @Component? 注解呀,寫的是類似于 @Service,@RestController,@Configuration? 等注解,不也是一樣可以被掃描到嗎?那這個 @Component 有什么特別的嗎?

元注解

在回答上面的問題之前,我們先來了解一下什么叫元注解,所謂元注解就是指一個能聲明在其他注解上的注解,換句話說就是如果一個注解被標注在其他注解上,那么它就是元注解。

要說明的是這個元注解并不是 Spring? 領域的東西, 而是 Java? 領域的,像 Java? 中的很多注解比如 @Document,@Repeatable? ,@Target 等都屬于元注解。

根據上面的解釋我們可以發現在 Spring? 容器里 @Component 

圖片

Configuration

圖片

controller

@Component 的派生性

通過上面的內容我們是不是可以猜測一下那就是 @Component? 注解的特性被"繼承"下來了?這就可以解釋為什么我們可以直接寫@Service,@RestController? 注解也是可以被掃描到的。但是由于 Java 的注解是不支持繼承的,比如你想通過下面的方式來實現注解的繼承是不合法的。

圖片

@interface

為了驗證我們的猜想,可以通過跟蹤源代碼來驗證一下,我們的目的是研究為什么不直接使用 @Component? 注解也能被 Spring? 掃描到,換句話說就是使用 @Service? 和 @RestController? 的注解也能成為 Spring Bean。

那我們很自然的就可以想到,在掃描的時候一定是根據注解來進行了判斷是否要初始化成 Spring Bean 的。我們只要找到了判斷條件就可以解決我們的疑惑了。

由于 SpringBoot? 項目是通過 main? 方法進行啟動的,調試起來還是很方便的,阿粉這邊準備了一個簡單的 SpringBoot? 工程,里面除了啟動類之外只有一個DemoController.java 代碼如下

package com.example.demojar.controller;

import org.springframework.web.bind.annotation.RestController;

@RestController
public class DemoController {
}

啟動類如下

package com.example.demojar;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.WebApplicationType;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.builder.SpringApplicationBuilder;
import org.springframework.context.ConfigurableApplicationContext;

@SpringBootApplication(scanBasePackages = {"com.example.demojar"})
public class DemoJarApplication {

public static void main(String[] args){
SpringApplication.run(DemoJarApplication.class, args);
}
}

Debug run? 方法,我們可以定位到 org.springframework.boot.SpringApplication#run(java.lang.String...) ?方法,該方法里面會初始化 SpringBoot? 上下文 context。

context = createApplicationContext();

默認情況下會進到下面的方法,并創建 AnnotationConfigServletWebServerApplicationContext? 并且其構造函數中構造了 ClassPathBeanDefinitionScanner 類路徑 Bean 掃描器。此處已經越來越接近掃描相關的內容了。

org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext.Factory#create

圖片

圖片

圖片

context? 上下文創建完成過后,接下來我們我們會接入到 org.springframework.context.support.AbstractApplicationContext#refresh?,再到 org.springframework.context.support.AbstractApplicationContext#invokeBeanFactoryPostProcessors

org.springframework.context.annotation.ConfigurationClassPostProcessor#postProcessBeanDefinitionRegistry

org.springframework.context.annotation.ConfigurationClassPostProcessor#processConfigBeanDefinitions

經過上面的步驟,最終可以可以定位到掃描的代碼在下面的方法 org.springframework.context.annotation.ComponentScanAnnotationParser#parse? 里面,調用前面上下文初始化的掃描器的 org.springframework.context.annotation.ClassPathBeanDefinitionScanner#doScan 方法,

圖片

圖片

到這里我們已經定位到了掃描具體包路徑的方法,這個方法里面主要看 findCandidateComponents(basePackage); 方法的內容,這個方法就是返回合法的候選組件。說明這個方法會最終返回需要被注冊成 Spring Bean 的候選組件,那我們重點就要看這個方法的實現。

跟蹤這個方法 org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider#findCandidateComponents? 進去我們可以看到通過加進類路徑里面的資源文件,然后再根據資源文件生成 MetadataReader? 對象,最后判斷這個 MetadataReader? 對象是否滿足候選組件的條件,如果滿足就添加到 Set 集合中進行返回。

圖片

繼續追蹤源碼我們可以找到具體的判斷方法在 org.springframework.core.type.filter.AnnotationTypeFilter#matchSelf? 方法中,如下所示,可以看到這里對 MetadataReader? 對象進行了判斷是否有元注解 @Component?。在調試的時候我們會發現 DemoController? 在此處會返回 true?,并且該 MetadataReader? 對象里面還有多個 mappings? ,其實這些 mappings? 對應的就是 Spring 的注解。

圖片

圖片

圖片

這個 mappings? 里面的注解確實包含了 @Component? 注解,因此會返回 true?。那么接下來問題就轉換成,我們的 DemoController? 對應的 MetadataReader 對象是如何創建的。

我們看回到 org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider#scanCandidateComponents? 方法,看看具體 MetadataReader 對象是如何創建的,MetadataReader metadataReader = getMetadataReaderFactory().getMetadataReader(resource);

圖片

圖片

通過構造方法 org.springframework.core.type.classreading.SimpleMetadataReader#SimpleMetadataReader? 進行 MetadataReader?  對象的創建,org.springframework.core.type.classreading.SimpleAnnotationMetadataReadingVisitor#visitEnd?,最終定位到 org.springframework.core.annotation.MergedAnnotationsCollection#MergedAnnotationsCollection? 這里進行 mappings 賦值。

圖片

繼續定位到 org.springframework.core.annotation.AnnotationTypeMappings.Cache#createMappings,org.springframework.core.annotation.AnnotationTypeMappings#addAllMappings,addAllmappings? 方法,內部使用了一個 while? 循環和 Deque? 來循環查詢元注解進行賦值,代碼如下所示,重點是這一行 Annotation[] metaAnnotations = AnnotationsScanner.getDeclaredAnnotations(source.getAnnotationType(), false);

private void addAllMappings(Class<? extends Annotation> annotationType,
Set<Class<? extends Annotation>> visitedAnnotationTypes){

Deque<AnnotationTypeMapping> queue = new ArrayDeque<>();
addIfPossible(queue, null, annotationType, null, visitedAnnotationTypes);
while (!queue.isEmpty()) {
AnnotationTypeMapping mapping = queue.removeFirst();
this.mappings.add(mapping);
addMetaAnnotationsToQueue(queue, mapping);
}
}

private void addMetaAnnotationsToQueue(Deque<AnnotationTypeMapping> queue, AnnotationTypeMapping source){
Annotation[] metaAnnotations = AnnotationsScanner.getDeclaredAnnotations(source.getAnnotationType(), false);
for (Annotation metaAnnotation : metaAnnotations) {
if (!isMappable(source, metaAnnotation)) {
continue;
}
Annotation[] repeatedAnnotations = this.repeatableContainers.findRepeatedAnnotations(metaAnnotation);
if (repeatedAnnotations != null) {
for (Annotation repeatedAnnotation : repeatedAnnotations) {
if (!isMappable(source, repeatedAnnotation)) {
continue;
}
addIfPossible(queue, source, repeatedAnnotation);
}
}
else {
addIfPossible(queue, source, metaAnnotation);
}
}
}

綜上所述我們可以發現盡管我們沒有直接寫 @Component? 注解,只要我們加了類似于 @Service,@RestController? 等注解也是可以成功被 Spring? 掃描到注冊成 Spring Bean? 的,本質的原因是因為這些注解底層都使用了 @Component? 作為元注解,經過源碼分析我們發現了只要有 @Component 元注解標注的注解類也是同樣會被進行掃描的。

總結

上面的源碼追蹤過程可能會比較枯燥和繁瑣,最后我們來簡單總結一下上面的內容:

  • 方法org.springframework.boot.SpringApplication#run(java.lang.String...) ?中進行 Spring 上下文的創建;
  • 在初始化上下文的時候會創建掃描器ClassPathBeanDefinitionScanner;
  • 在org.springframework.context.support.AbstractApplicationContext#refresh? 進行 beanFactory 準備;
  • org.springframework.context.annotation.ClassPathBeanDefinitionScanner#doScan 進行資源掃描
  • 在org.springframework.core.annotation.MergedAnnotationsCollection#MergedAnnotationsCollection? 進行注解 mappings 的賦值;
  • org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider#scanCandidateComponents 方法中進行候選組件的判斷;

上面追蹤的過程可能會比較復雜,但是只要我們理解了原理還是可以慢慢跟上的,因為我們只要把握好了方向,知道首先肯定會進行資源掃描,掃描完了肯定是根據注解之間的關系進行判斷,最終得到我們需要的候選組件集合。至于如何創建 MetadataReader 和如何獲取元注解,只要我們一步步看下去就是可以找到的。

最后說明一下,Spring Framework? 每個版本的具體實現會有差異,阿粉使用的版本是 5.3.24 ,所以如果小伙伴看到自己的代碼追蹤的效果跟阿粉的不一樣也不會奇怪,可能是因為版本不一樣而已,不過本質上都是一樣的。

責任編輯:武曉燕 來源: Java極客技術
相關推薦

2022-09-28 18:16:34

JavaJDK

2023-12-20 08:23:53

NIO組件非阻塞

2015-12-01 13:33:51

UnikernelLinux運維

2021-11-12 05:59:23

容災備份5G

2016-09-29 15:43:33

2022-11-28 00:04:17

2024-01-15 12:16:37

2019-03-14 12:39:55

安全云計算深信服

2020-09-03 06:42:12

線程安全CPU

2024-07-30 08:22:47

API前端網關

2021-11-09 09:39:21

路由器硬件設備網絡

2024-11-08 09:48:38

異步編程I/O密集

2024-07-01 08:40:18

tokio派生線程

2023-06-30 08:26:24

Java注解Java程序元素

2024-02-19 07:44:52

虛擬機Java平臺

2024-03-19 08:01:54

服務熔斷軟件設計模式微服務

2023-07-11 00:12:05

2024-06-27 10:51:28

生成式AI領域

2024-06-11 09:02:30

2023-01-04 11:39:45

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 99亚洲精品 | 久久久久国产一区二区三区四区 | 国产伦一区二区三区视频 | 色网在线观看 | 在线日韩精品视频 | 拍真实国产伦偷精品 | 成人h动漫亚洲一区二区 | 国产精品毛片av | 久久精品免费 | 精品av天堂毛片久久久借种 | 欧美一级做性受免费大片免费 | 亚洲精品久久区二区三区蜜桃臀 | 亚洲首页| 国产视频三区 | 亚洲第一黄色网 | 国产免费xxx| 成人黄色av| 欧美在线观看一区 | 国产精品久久久久久久久久软件 | 亚洲精品粉嫩美女一区 | 久久久av | 91精品久久久久久久久中文字幕 | 亚洲欧美视频 | 91精品国产综合久久国产大片 | 91精品国产乱码久久久久久久久 | 日本久久www成人免 成人久久久久 | 国产高清免费在线 | 日韩精品视频在线观看一区二区三区 | 欧美日韩在线免费 | 国产小视频在线 | 久久久久久久国产 | 欧美精品一区三区 | 中文字幕人成乱码在线观看 | 久久久无码精品亚洲日韩按摩 | 久久久久久美女 | 国产精品一区二区日韩 | 国产成人精品免费 | 国产成人精品一区二区 | 精品久久国产老人久久综合 | 日本高清精品 | 伦理午夜电影免费观看 |