欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Autowired的注入過程源碼解析

 更新時(shí)間:2022年09月03日 14:34:26   作者:楊同學(xué)_  
這篇文章主要為大家介紹了Autowired的注入過程源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

一、案例場(chǎng)景

在使用 @Autowired 時(shí),你或多或少都會(huì)遇過類似的錯(cuò)誤:

required a single bean, but 2 were found

為了重現(xiàn)這個(gè)錯(cuò)誤,我們可以先寫一個(gè)案例來模擬下。

@RestController
@Slf4j
@Validated
public class StudentController {
    @Autowired
    DataService dataService;
    @RequestMapping(path = "students/{id}", method = RequestMethod.DELETE)
    public void deleteStudent(@PathVariable("id") @Range(min = 1,max = 100) int id) {
        dataService.deleteStudent(id);
    }
}

其中 DataService 是一個(gè)接口,其實(shí)現(xiàn)依托于 Oracle,代碼示意如下:

public interface DataService {
    void deleteStudent(int id);
} 
@Repository
@Slf4j
public class OracleDataService implements DataService {
    @Override
    public void deleteStudent(int id) {
        log.info("delete student info maintained by oracle");
    }
}

截止目前,運(yùn)行并測(cè)試程序是毫無問題的。直到某天,我們接到節(jié)約成本的需求,希望把一些部分非核心的業(yè)務(wù)從 Oracle 遷移到社區(qū)版 Cassandra,所以我們自然會(huì)先添加上一個(gè)新的 DataService 實(shí)現(xiàn),代碼如下:

@Repository
@Slf4j
public class CassandraDataService implements DataService{
    @Override
    public void deleteStudent(int id) {
        log.info("delete student info maintained by cassandra");
    }
}

此時(shí),程序就已經(jīng)無法啟動(dòng)了,報(bào)錯(cuò)如下:

image-20220719210803377

二、案例解析

首先,我們先來了解下 @Autowired 發(fā)生的位置和核心過程。當(dāng)一個(gè) Bean 被構(gòu)建時(shí),核心包括兩個(gè)基本步驟:

  • 執(zhí)行 AbstractAutowireCapableBeanFactory#createBeanInstance 方法:通過構(gòu)造器反射構(gòu)造出這個(gè) Bean,在此案例中相當(dāng)于構(gòu)建出 StudentController 的實(shí)例;
  • 執(zhí)行 AbstractAutowireCapableBeanFactory#populateBean 方法:填充(即設(shè)置)這個(gè) Bean,在本案例中,相當(dāng)于設(shè)置 StudentController 實(shí)例中被 @Autowired 標(biāo)記的 dataService 屬性成員。

在步驟 2 中,“填充”過程的關(guān)鍵就是執(zhí)行各種 BeanPostProcessor 處理器,關(guān)鍵代碼如下:

protected void populateBean(String beanName, RootBeanDefinition mbd, @Nullable BeanWrapper bw) {
  //省略非關(guān)鍵代碼
  for (BeanPostProcessor bp : getBeanPostProcessors()) {
    if (bp instanceof InstantiationAwareBeanPostProcessor) {
      InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;
      PropertyValues pvsToUse = ibp.postProcessProperties(pvs, bw.getWrappedInstance(), beanName);
      //省略非關(guān)鍵代碼
    }
  }
}

在上述代碼執(zhí)行過程中,因?yàn)?StudentController 含有標(biāo)記為 Autowired 的成員屬性 dataService,所以會(huì)使用到AutowiredAnnotationBeanPostProcessor(BeanPostProcessor 中的一種)來完成“裝配”過程:找出合適的 DataService 的 bean 并設(shè)置給StudentController#dataService。如果深究這個(gè)裝配過程,又可以細(xì)分為兩個(gè)步驟:

  • 尋找出所有需要依賴注入的字段和方法,參考 AutowiredAnnotationBeanPostProcessor#postProcessProperties 中的代碼行:
InjectionMetadata metadata = findAutowiringMetadata(beanName, bean.getClass(), pvs);
  • 根據(jù)依賴信息尋找出依賴并完成注入,以字段注入為例,參考 AutowiredFieldElement#inject 方法:
@Override
protected void inject(Object bean, @Nullable String beanName, @Nullable PropertyValues pvs) throws Throwable {
    Field field = (Field) this.member;
    Object value;
    //省略非關(guān)鍵代碼
    DependencyDescriptor desc = new DependencyDescriptor(field, this.required);
    //尋找“依賴”,desc為"dataService"的DependencyDescriptor
    value = beanFactory.resolveDependency(desc, beanName, autowiredBeanNames, typeConverter);
    //省略非關(guān)鍵代碼
    if (value != null) {
        ReflectionUtils.makeAccessible(field);
        //裝配“依賴”
        field.set(bean, value);
    }
}

當(dāng)我們根據(jù) DataService 這個(gè)類型來找出依賴時(shí),我們會(huì)找出 2 個(gè)依賴,分別為 CassandraDataService 和 OracleDataService。在這樣的情況下,如果同時(shí)滿足以下兩個(gè)條件則會(huì)拋出本案例的錯(cuò)誤:

  • 調(diào)用 determineAutowireCandidate 方法來選出優(yōu)先級(jí)最高的依賴,但是發(fā)現(xiàn)并沒有優(yōu)先級(jí)可依據(jù)。具體選擇過程可參考DefaultListableBeanFactory#determineAutowireCandidate。

優(yōu)先級(jí)的決策是先根據(jù) @Primary 來決策,其次是 @Priority 決策,最后是根據(jù) Bean 名字的嚴(yán)格匹配來決策。如果這些幫助決策優(yōu)先級(jí)的注解都沒有被使用,名字也不精確匹配,則返回 null,告知無法決策出哪種最合適。

  • @Autowired 要求是必須注入的(即 required 保持默認(rèn)值為 true),或者注解的屬性類型并不是可以接受多個(gè) Bean 的類型,例如數(shù)組、Map、集合。這點(diǎn)可以參考 DefaultListableBeanFactory#indicatesMultipleBeans 的實(shí)現(xiàn)。

三、問題修正

第一,我們可以通過使用標(biāo)記 @Primary 的方式來讓被標(biāo)記的候選者有更高優(yōu)先級(jí),從而避免報(bào)錯(cuò)。

@Repository
@Primary
@Slf4j
public class OracleDataService implements DataService{
  //省略非關(guān)鍵代碼
}

但是這種方式并不一定符合業(yè)務(wù)需求。

第二,我們可以使用下面的方式去修改:

@Autowired
DataService oracleDataService;

將屬性名和 Bean 名字精確匹配,這樣就可以讓注入選擇不犯難:需要 Oracle 時(shí)指定屬性名為 oracleDataService,需要 Cassandra 時(shí)則指定屬性名為 cassandraDataService。

第三,還可以采用 @Qualifier 來顯式指定引用的是那種服務(wù),例如采用下面的方式:

@Autowired
@Qualifier("cassandraDataService")
DataService dataService;

這種方式之所以能解決問題,在于它能讓尋找出的 Bean 只有一個(gè)(即精確匹配),所以壓根不會(huì)出現(xiàn)后面的決策過程,可以參考 DefaultListableBeanFactory#doResolveDependency。

public Object doResolveDependency(DependencyDescriptor descriptor, String beanName,
		Set<String> autowiredBeanNames, TypeConverter typeConverter) throws BeansException {
    //...
    Object value = getAutowireCandidateResolver().getSuggestedValue(descriptor);
    //...
}

以上就是Autowired的注入過程源碼解析的詳細(xì)內(nèi)容,更多關(guān)于Autowired注入過程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • SpringBoot+Idea熱部署實(shí)現(xiàn)流程解析

    SpringBoot+Idea熱部署實(shí)現(xiàn)流程解析

    這篇文章主要介紹了SpringBoot+Idea熱部署實(shí)現(xiàn)流程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-11-11
  • java實(shí)現(xiàn)String字符串處理各種類型轉(zhuǎn)換

    java實(shí)現(xiàn)String字符串處理各種類型轉(zhuǎn)換

    在日常的程序開發(fā)中,經(jīng)常會(huì)涉及到不同類型之間的轉(zhuǎn)換,本文主要介紹了String字符串處理各種類型轉(zhuǎn)換,具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-10-10
  • Java之JSF框架案例詳解

    Java之JSF框架案例詳解

    這篇文章主要介紹了Java之JSF框架案例詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-09-09
  • 淺談Maven安裝及環(huán)境配置出錯(cuò)的解決辦法

    淺談Maven安裝及環(huán)境配置出錯(cuò)的解決辦法

    這篇文章主要介紹了淺談Maven安裝及環(huán)境配置出錯(cuò)的解決辦法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-09-09
  • 詳解spring boot使用@Retryable來進(jìn)行重處理

    詳解spring boot使用@Retryable來進(jìn)行重處理

    本篇文章主要介紹了詳解spring boot使用@Retryable來進(jìn)行重處理,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • Java Spring5學(xué)習(xí)之JdbcTemplate詳解

    Java Spring5學(xué)習(xí)之JdbcTemplate詳解

    這篇文章主要介紹了Java Spring5學(xué)習(xí)之JdbcTemplate詳解,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴們有非常好的幫助,需要的朋友可以參考下
    2021-05-05
  • 詳解spring boot 以jar的方式啟動(dòng)常用shell腳本

    詳解spring boot 以jar的方式啟動(dòng)常用shell腳本

    本篇文章主要介紹了詳解spring boot 以jar的方式啟動(dòng)常用shell腳本,具有一定的參考價(jià)值,有興趣的可以了解一下
    2017-09-09
  • Spring之什么是ObjectFactory?什么是ObjectProvider?

    Spring之什么是ObjectFactory?什么是ObjectProvider?

    這篇文章主要介紹了Spring之什么是ObjectFactory?什么是ObjectProvider?具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-01-01
  • 淺談JAVA 異常對(duì)于性能的影響

    淺談JAVA 異常對(duì)于性能的影響

    Java的異常處理為什么會(huì)影響性能?異常開銷很大。那么,這是不是就意味著您不該使用異常?當(dāng)然不是。但是,何時(shí)應(yīng)該使用異常,何時(shí)又不應(yīng)該使用異常呢?不幸的是,答案不是一下子就說得清楚的,我們來詳細(xì)探討下。
    2015-05-05
  • java 如何讀取遠(yuǎn)程主機(jī)文件

    java 如何讀取遠(yuǎn)程主機(jī)文件

    這篇文章主要介紹了java 如何讀取遠(yuǎn)程主機(jī)文件的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-02-02

最新評(píng)論