Autowired的注入過程源碼解析
一、案例場景
在使用 @Autowired 時,你或多或少都會遇過類似的錯誤:
required a single bean, but 2 were found
為了重現(xiàn)這個錯誤,我們可以先寫一個案例來模擬下。
@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 是一個接口,其實現(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");
}
}
截止目前,運行并測試程序是毫無問題的。直到某天,我們接到節(jié)約成本的需求,希望把一些部分非核心的業(yè)務從 Oracle 遷移到社區(qū)版 Cassandra,所以我們自然會先添加上一個新的 DataService 實現(xiàn),代碼如下:
@Repository
@Slf4j
public class CassandraDataService implements DataService{
@Override
public void deleteStudent(int id) {
log.info("delete student info maintained by cassandra");
}
}
此時,程序就已經(jīng)無法啟動了,報錯如下:
二、案例解析
首先,我們先來了解下 @Autowired 發(fā)生的位置和核心過程。當一個 Bean 被構建時,核心包括兩個基本步驟:
- 執(zhí)行 AbstractAutowireCapableBeanFactory#createBeanInstance 方法:通過構造器反射構造出這個 Bean,在此案例中相當于構建出 StudentController 的實例;
- 執(zhí)行 AbstractAutowireCapableBeanFactory#populateBean 方法:填充(即設置)這個 Bean,在本案例中,相當于設置 StudentController 實例中被 @Autowired 標記的 dataService 屬性成員。
在步驟 2 中,“填充”過程的關鍵就是執(zhí)行各種 BeanPostProcessor 處理器,關鍵代碼如下:
protected void populateBean(String beanName, RootBeanDefinition mbd, @Nullable BeanWrapper bw) {
//省略非關鍵代碼
for (BeanPostProcessor bp : getBeanPostProcessors()) {
if (bp instanceof InstantiationAwareBeanPostProcessor) {
InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;
PropertyValues pvsToUse = ibp.postProcessProperties(pvs, bw.getWrappedInstance(), beanName);
//省略非關鍵代碼
}
}
}
在上述代碼執(zhí)行過程中,因為 StudentController 含有標記為 Autowired 的成員屬性 dataService,所以會使用到AutowiredAnnotationBeanPostProcessor(BeanPostProcessor 中的一種)來完成“裝配”過程:找出合適的 DataService 的 bean 并設置給StudentController#dataService。如果深究這個裝配過程,又可以細分為兩個步驟:
- 尋找出所有需要依賴注入的字段和方法,參考 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;
//省略非關鍵代碼
DependencyDescriptor desc = new DependencyDescriptor(field, this.required);
//尋找“依賴”,desc為"dataService"的DependencyDescriptor
value = beanFactory.resolveDependency(desc, beanName, autowiredBeanNames, typeConverter);
//省略非關鍵代碼
if (value != null) {
ReflectionUtils.makeAccessible(field);
//裝配“依賴”
field.set(bean, value);
}
}
當我們根據(jù) DataService 這個類型來找出依賴時,我們會找出 2 個依賴,分別為 CassandraDataService 和 OracleDataService。在這樣的情況下,如果同時滿足以下兩個條件則會拋出本案例的錯誤:
- 調(diào)用 determineAutowireCandidate 方法來選出優(yōu)先級最高的依賴,但是發(fā)現(xiàn)并沒有優(yōu)先級可依據(jù)。具體選擇過程可參考DefaultListableBeanFactory#determineAutowireCandidate。
優(yōu)先級的決策是先根據(jù) @Primary 來決策,其次是 @Priority 決策,最后是根據(jù) Bean 名字的嚴格匹配來決策。如果這些幫助決策優(yōu)先級的注解都沒有被使用,名字也不精確匹配,則返回 null,告知無法決策出哪種最合適。
- @Autowired 要求是必須注入的(即 required 保持默認值為 true),或者注解的屬性類型并不是可以接受多個 Bean 的類型,例如數(shù)組、Map、集合。這點可以參考 DefaultListableBeanFactory#indicatesMultipleBeans 的實現(xiàn)。
三、問題修正
第一,我們可以通過使用標記 @Primary 的方式來讓被標記的候選者有更高優(yōu)先級,從而避免報錯。
@Repository
@Primary
@Slf4j
public class OracleDataService implements DataService{
//省略非關鍵代碼
}
但是這種方式并不一定符合業(yè)務需求。
第二,我們可以使用下面的方式去修改:
@Autowired DataService oracleDataService;
將屬性名和 Bean 名字精確匹配,這樣就可以讓注入選擇不犯難:需要 Oracle 時指定屬性名為 oracleDataService,需要 Cassandra 時則指定屬性名為 cassandraDataService。
第三,還可以采用 @Qualifier 來顯式指定引用的是那種服務,例如采用下面的方式:
@Autowired
@Qualifier("cassandraDataService")
DataService dataService;
這種方式之所以能解決問題,在于它能讓尋找出的 Bean 只有一個(即精確匹配),所以壓根不會出現(xiàn)后面的決策過程,可以參考 DefaultListableBeanFactory#doResolveDependency。
public Object doResolveDependency(DependencyDescriptor descriptor, String beanName,
Set<String> autowiredBeanNames, TypeConverter typeConverter) throws BeansException {
//...
Object value = getAutowireCandidateResolver().getSuggestedValue(descriptor);
//...
}以上就是Autowired的注入過程源碼解析的詳細內(nèi)容,更多關于Autowired注入過程的資料請關注腳本之家其它相關文章!
相關文章
SpringBoot+Idea熱部署實現(xiàn)流程解析
這篇文章主要介紹了SpringBoot+Idea熱部署實現(xiàn)流程解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-11-11
java實現(xiàn)String字符串處理各種類型轉(zhuǎn)換
在日常的程序開發(fā)中,經(jīng)常會涉及到不同類型之間的轉(zhuǎn)換,本文主要介紹了String字符串處理各種類型轉(zhuǎn)換,具有一定的參考價值,感興趣的可以了解一下2023-10-10
詳解spring boot使用@Retryable來進行重處理
本篇文章主要介紹了詳解spring boot使用@Retryable來進行重處理,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-06-06
詳解spring boot 以jar的方式啟動常用shell腳本
本篇文章主要介紹了詳解spring boot 以jar的方式啟動常用shell腳本,具有一定的參考價值,有興趣的可以了解一下2017-09-09
Spring之什么是ObjectFactory?什么是ObjectProvider?
這篇文章主要介紹了Spring之什么是ObjectFactory?什么是ObjectProvider?具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-01-01

