spring的自動裝配,騷話@Autowired的底層工作原理

来源:https://www.cnblogs.com/youzhibing/archive/2019/06/21/11031216.html
-Advertisement-
Play Games

前言 開心一刻 十年前,我:我交女票了,比我大兩歲。媽:不行!趕緊分! 八年前,我:我交女票了,個比我小兩歲,外地的。媽:你就不能讓我省點心? 五年前,我:我交女票了,市長的女兒。媽:別人還能看上你?分了吧! 今年,我挺著大肚子踏進家門。媽:閨女啊,你終於開竅了 ! 前情回顧 Spring拓展介面之 ...


前言

  開心一刻

    十年前,我:我交女票了,比我大兩歲。媽:不行!趕緊分!
    八年前,我:我交女票了,比我小兩歲,外地的。媽:你就不能讓我省點心?
    五年前,我:我交女票了,市長的女兒。媽:別人還能看上你?分了吧!
    今年,我挺著大肚子踏進家門。媽:閨女啊,你終於開竅了 !

前情回顧

  Spring拓展介面之BeanPostProcessor,我們來看看它的底層實現中講到了spring對BeanPostProcessor的底層支持,並且知道了BeanPostProcessor的兩個方法:postProcessBeforeInitialization、postProcessAfterInitialization的執行時機,沒看的小伙伴可以回過頭去看看。本來spring的自動裝配是打算放到上一篇博文中詳細講解的,可後來覺得篇幅可能太大了(細心的小伙伴可能會有這樣的表情:,除了幾幅圖,真沒什麼內容!),既然你們都感覺出來了,那我也就明人不說暗話了,之所以沒放到上篇講解,確實是因為篇幅太大了(哈哈哈,是不是很想打我? ); 好了,我們言歸正傳,之所以沒放到上篇來講,篇幅只是原因之一,最主要的原因是發現我犯錯了! 犯什麼錯了呢(不是黃賭毒啊,那是犯罪,我是正人君子!),我想當然了! 理所當然的認為自動裝配是在AutowiredAnnotationBeanPostProcessor的postProcessBeforeInitialization或postProcessAfterInitialization中實現的,我們來看下AutowiredAnnotationBeanPostProcessor類繼承圖

  它間接實現了BeanPostProcessor,我們再去看下那兩個方法(在父類InstantiationAwareBeanPostProcessorAdapter中)

@Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
    return bean;
}

@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
    return bean;
}
View Code

  竟然啥也沒乾,只是簡單的return bean; 當自己深以為然的認知被推翻時,那感覺真是斃了狗了 所以自動裝配不能和BeanPostProcessor放一塊講,不得不開兩篇來分開講,我們都知道:強扭的瓜不甜!

自動裝配簡單示例

  我們先來看一個簡單的自動裝配的示例,完整實例代碼:spring-boot-BeanPostProcessor

  AnimalConfig

View Code

  AnimalServiceImpl

@Service
public class AnimalServiceImpl implements IAnimalService {

    @Autowired
    private Dog dog;
    @Resource
    private Cat cat;
    @Inject
    private Pig pig;

    @Override
    public void printName() {
        System.out.println(dog.getName());
        System.out.println(cat.getName());
        System.out.println(pig.getName());
    }
}
View Code

  AnimalTest

@RunWith(SpringRunner.class)
@SpringBootTest(classes={Application.class})
public class AnimalTest {

    @Autowired
    private IAnimalService animalService;

    @Test
    public void test() {
        animalService.printName();
    }
}
View Code

  運行結果

  我們在AnimalConfig中只是將Dog、Cat、Pig的實例註冊到了spring容器,那為什麼AnimalServiceImpl實例能夠直接應用這些實例了,我們並沒有手動的將這些實例賦值到AnimalServiceImpl實例呀? 這其實就是spring提供的自動裝配功能,雖然我們沒有手動的將這些實例賦值到AnimalServiceImpl實例,但是我們發現AnimalServiceImpl的屬性實例上多了一些註解:@Autowired、@Resource、@Inject,spring通過這些註解自動完成了屬性實例的註入,而不需要我們手動的去賦值了;那麼spring是如何實現自動裝配的呢? 我們慢慢往下看(註意:後文主要以@Autowired為例來講解

自動裝配源碼解析

  AutowiredAnnotationBeanPostProcessor的實例化與註冊

    不管怎麼說,AutowiredAnnotationBeanPostProcessor終歸還是一個BeanPostProcessor,那麼它的實例化與註冊(註冊到spring的beanFactory)過程與BeanPostProcessor的實例化與註冊一樣,在spring的啟動過程中,刷新上下文(refresh)的時候,會調用registerBeanPostProcessors(beanFactory)方法完成BeanPostProcessor的實例化與註冊,後續再調用finishBeanFactoryInitialization(beanFactory)實例化非延遲載入的單例bean時,會用到上述註冊的BeanPostProcessor

    AutowiredAnnotationBeanPostProcessor的構造方法值得我們看看

public AutowiredAnnotationBeanPostProcessor() {
    this.autowiredAnnotationTypes.add(Autowired.class);
    this.autowiredAnnotationTypes.add(Value.class);
    try {
        this.autowiredAnnotationTypes.add((Class<? extends Annotation>)
                ClassUtils.forName("javax.inject.Inject", AutowiredAnnotationBeanPostProcessor.class.getClassLoader()));
        logger.info("JSR-330 'javax.inject.Inject' annotation found and supported for autowiring");
    }
    catch (ClassNotFoundException ex) {
        // JSR-330 API not available - simply skip.
    }
}
View Code

    預設情況下,AutowiredAnnotationBeanPostProcessor支持@Autowired和@Value,如果類路徑下有java.inject.Inject(也就是引入了javax.inject.jar),那麼也支持@Inject註解,是不是與我們最初的認知有些不一樣?。將支持的註解放到了autowiredAnnotationTypes屬性中,後續會用到該屬性

  bean的實例化與依賴註入

    預設情況下,spring會把spring容器中的bean當成non-lazy-init singleton來處理(有些特殊的bean除外),也就是說會在spring的啟動過程中就會逐個實例化這些bean,並對這些bean進行依賴註入;當我們真正用到這些bean的時候,直接用就行,不用再去實例化,也不用再去註入bean的相關依賴,spring是不是很厲害?。具體是不是說的這樣,大家準備好花生、瓜子和啤酒,好戲即將開始

    我們先找到正確的入口,然後用下圖省略掉無聊的前戲,直接進入高潮:doCreateBean(不應該是這個嗎,一天天的盡胡思亂想

    doCreateBean內容如下

protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args)
        throws BeanCreationException {

    // Instantiate the bean.
    BeanWrapper instanceWrapper = null;
    if (mbd.isSingleton()) {
        instanceWrapper = this.factoryBeanInstanceCache.remove(beanName);
    }
    if (instanceWrapper == null) {
        // 創建bean實例
        instanceWrapper = createBeanInstance(beanName, mbd, args);
    }
    final Object bean = instanceWrapper.getWrappedInstance();
    Class<?> beanType = instanceWrapper.getWrappedClass();
    if (beanType != NullBean.class) {
        mbd.resolvedTargetType = beanType;
    }

    // Allow post-processors to modify the merged bean definition.
    // 允許後置處理器來修改bean定義
    synchronized (mbd.postProcessingLock) {
        if (!mbd.postProcessed) {
            try {
                // 調用MergedBeanDefinitionPostProcessor的postProcessMergedBeanDefinition方法
                // AutowiredAnnotationBeanPostProcessor實現了MergedBeanDefinitionPostProcessor,即MergedBeanDefinitionPostProcessor的MergedBeanDefinitionPostProcessor會被調用
                applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);
            }
            catch (Throwable ex) {
                throw new BeanCreationException(mbd.getResourceDescription(), beanName,
                        "Post-processing of merged bean definition failed", ex);
            }
            mbd.postProcessed = true;
        }
    }

    // Eagerly cache singletons to be able to resolve circular references 立即緩存單例以便能夠解析迴圈引用
    // even when triggered by lifecycle interfaces like BeanFactoryAware.
    boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
            isSingletonCurrentlyInCreation(beanName));
    if (earlySingletonExposure) {
        if (logger.isDebugEnabled()) {
            logger.debug("Eagerly caching bean '" + beanName +
                    "' to allow for resolving potential circular references");
        }
        addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
    }

    // Initialize the bean instance.
    Object exposedObject = bean;
    try {
        // 填充bean,包含依賴註入
        populateBean(beanName, mbd, instanceWrapper);
        // 初始化bean,BeanPostProcessor的兩個方法在此中被調用
        exposedObject = initializeBean(beanName, exposedObject, mbd);
    }
    catch (Throwable ex) {
        if (ex instanceof BeanCreationException && beanName.equals(((BeanCreationException) ex).getBeanName())) {
            throw (BeanCreationException) ex;
        }
        else {
            throw new BeanCreationException(
                    mbd.getResourceDescription(), beanName, "Initialization of bean failed", ex);
        }
    }

    if (earlySingletonExposure) {
        Object earlySingletonReference = getSingleton(beanName, false);
        if (earlySingletonReference != null) {
            if (exposedObject == bean) {
                exposedObject = earlySingletonReference;
            }
            else if (!this.allowRawInjectionDespiteWrapping && hasDependentBean(beanName)) {
                String[] dependentBeans = getDependentBeans(beanName);
                Set<String> actualDependentBeans = new LinkedHashSet<>(dependentBeans.length);
                for (String dependentBean : dependentBeans) {
                    if (!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)) {
                        actualDependentBeans.add(dependentBean);
                    }
                }
                if (!actualDependentBeans.isEmpty()) {
                    throw new BeanCurrentlyInCreationException(beanName,
                            "Bean with name '" + beanName + "' has been injected into other beans [" +
                            StringUtils.collectionToCommaDelimitedString(actualDependentBeans) +
                            "] in its raw version as part of a circular reference, but has eventually been " +
                            "wrapped. This means that said other beans do not use the final version of the " +
                            "bean. This is often the result of over-eager type matching - consider using " +
                            "'getBeanNamesOfType' with the 'allowEagerInit' flag turned off, for example.");
                }
            }
        }
    }

    // Register bean as disposable.
    try {
        registerDisposableBeanIfNecessary(beanName, bean, mbd);
    }
    catch (BeanDefinitionValidationException ex) {
        throw new BeanCreationException(
                mbd.getResourceDescription(), beanName, "Invalid destruction signature", ex);
    }

    return exposedObject;
}
View Code

    我們重點看下posProcessMergedBeanDefinition方法和populateBean方法

  posProcessMergedBeanDefinition

    可以看到會讀取bean的field和method上的註解,並判斷該註解是否在autowiredAnnotationTypes中,如果在則將field封裝成AutowiredFiledElement對象、將method封裝成AutoWiredMethodElement對象,並存放到InjectionMetadata對象的Set<InjectedElement> checkedElements屬性中,最後將該InjectionMetadata對象緩存到了AutowiredAnnotationBeanPostProcessor的Map<String, InjectionMetadata> injectionMetadataCache屬性中;說白了就是將bean中被@Autowried(當然還包括@Value、@Inject)修飾的field、method找出來,封裝成InjectionMetadata對象並緩存起來,就這麼簡單。不僅僅是上圖中的animalServiceImpl這一個bean,spring中所有的非延遲載入的bean都會走這個創建流程。是不是很簡單,是不是幹勁十足了

  populateBean

    調用AutowiredAnnotationBeanPostProcessor的postProcessPropertyValues方法,從injectionMetadataCache中獲取當前bean的依賴信息,比如animalServiceImpl依賴的dog、pig(有人可能會有這樣的疑問:cat呢? cat是被@Resource修飾的,而@Resource不是由AutowiredAnnotationBeanPostProcessor支持,後續會講由誰支持),然後逐個將依賴bean註入到目標bean(將dog、pig實例註入到animalServiceImpl實例中);依賴bean從哪來呢?還是從beanFactory中獲取,如果不存在,則又回到bean的創建過程把依賴bean(dog、pig)創建出來,流程與創建animalServiceImpl實例一模一樣,也就說在animalServiceImpl實例的依賴註入過程中會把dog、pig對象也創建出來,而不是等到spring逐個實例化bean的過程中輪到dog、pig才實例化dog、pig,那後續輪到dog、pig時怎麼辦了,spring會把創建的bean緩存起來,下次就直接從緩存中取了。上圖只演示Field的,Method也差不太多,就不演示了,都是通過反射實現的 。

總結

  1、bean的創建與初始化

    (1)instanceWrapper = createBeanInstance(beanName, mbd, args)  創建目標bean實例;

    (2)applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName)  尋找目標bean的依賴;

    (3)populateBean(beanName, mbd, instanceWrapper)  填充目標bean,完成依賴註入; (這裡的迴圈依賴,有興趣的可以自行去琢磨下)

    (4)initializeBean(beanName, exposedObject, mbd)  初始化目標bean

  2、自動裝配與自動配置

    自動配置一般而言說的是spring的@Autowired,是spring的特性之一,而自動配置是springboot的@Configuration,是springboot的特性之一

  3、Spring支持幾下幾種自動裝配的註解

    @Autowired、@Inject、@Resource以及@Value,用的最多的應該是@Autowired(至少我是這樣的),@Inject和@Value也是由AutowiredAnnotationBeanPostProcessor支持,而@Resource是由CommonAnnotationBeanPostProcessor支持(還支持@PostConstruct、@PreDestroy等註解)

    關於@Value與@Autowired,不知道大家是否清楚他們之間的區別,不清楚的可以看看:Spring: @Value vs. @Autowired或者spring的官方文檔,總結下:@Value >= @Autowired,只是平時應用中,@Value更多的是用來註入配置值(如:@Value("${db.url}")),而@Autowired則是bean對象的註入

參考

  JAVA 註解的基本原理

  深入理解Spring系列之十四:@Autowired是如何工作的


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一. 現狀 由於之前採用Lodop列印控制項(商業版付費,可以使用免費版 但是會有水印)去列印小票,是一行一行的列印,但是不滿足UI給到複雜佈局的小票樣式,所以得重新考慮如何來實現。 二. 介紹 art template介紹 art template 是一個簡約、超快的模板引擎。 它採用作用域預聲明的 ...
  • 主要記錄在ios瀏覽器出現觸摸無限載入的情況 使用vue-scroller和mescroll.js/mescroll.vue先踩ios瀏覽器預設滑動會影響mescroll的方法調用。 首先給公共js加入以下代碼禁用我們的頁面在ios瀏覽器下會滑動上下頁面。 下麵給body設置樣式: html, bo ...
  • 父組件主動獲取子組件的數據和方法 1.調用子組件的時候 定義一個ref <headerchild ref="headerChild"></headerchild> 2.在父組件裡面通過 this.$refs.headerChild.屬性 t his.$refs.headerChild.方法 子組件主 ...
  • 1、安裝charles,點擊幫助——ssl代理——在移動設備或遠程瀏覽器上安裝charles root證書,看到如下界面: 2、在手機保證和電腦連接同一個wifi的前提下,開啟手機代理,輸入伺服器地址:192.168.5.252,埠號為:8888,有時候新手機連接代理,charles會提示是否允許 ...
  • 概述 最近在學習netty的相關知識,也在看netty的源碼,光看不練假把式,所以也正好利用自己學習的機會寫幾篇netty的分析文章,主要還是一些源碼解析的文章,一方面有輸出會促使自己在看源碼,學習原理的過程中更系統,更深入,同時也能加強記憶,鞏固對知識的理解。 關於netty的簡介和應用我就不做介 ...
  • 1.SOA SOA(Service-Oriented Architecture)面向服務架構,將應用程式不同功能單元(稱為服務)進行拆分,並通過這些服務之間定義良好的介面和契約聯繫起來。 SOA 不是特定的規範,是一種技術思想,一種理念,上圖為 SOA 架構的參考模型。 SOA 是一種粗粒度、松耦合 ...
  • 實驗三、數據挖掘之決策樹 一、實驗目的 1. 熟悉掌握決策樹的原理, 2. 熟練掌握決策樹的生成方法與過程 二、實驗工具 1. Anaconda 2. sklearn 3. pydotplus 三、實驗簡介 決策樹是一個非參數的監督式學習方法,主要用於分類和回歸。演算法的目標是通過推斷數據特征,學習決 ...
  • 實驗一、數據處理之Numpy 一、實驗目的 1. 瞭解numpy庫的基本功能 2. 掌握Numpy庫的對數組的操作與運算 二、實驗工具: 1. Anaconda 2. Numpy 三、Numpy簡介 Numpy 的英文全稱為 Numerical Python,指Python 面向數值計算的第三方庫。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...