版權說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權,請進行舉報或認領
文檔簡介
第Spring源碼解析Bean屬性填充目錄前言屬性填充執(zhí)行回調(diào)方法及后置處理器
前言
在上一篇文章中,我們分析了Spring中Bean的實例化過程,在結尾我們知道了雖然bean的實例化完成了,但是其中的屬性還沒有被注入,今天我們就接著來分析屬性是如何被注入的。
屬性填充
實例化完成后,回到上面第3條的doCreateBean方法中,看一下用BeanWrapper產(chǎn)生的原生對象,里面dao這個屬性還是null值。
回歸一下之前的代碼,接下來要調(diào)用populateBean方法進行屬性的填充:
ObjectexposedObject=bean;
try{
populateBean(beanName,mbd,instanceWrapper);
exposedObject=initializeBean(beanName,exposedObject,mbd);
}
看一下populateBean中的核心代碼:
for(BeanPostProcessorbp:getBeanPostProcessors()){
if(bpinstanceofInstantiationAwareBeanPostProcessor){
InstantiationAwareBeanPostProcessoribp=(InstantiationAwareBeanPostProcessor)bp;
pvs=ibp.postProcessPropertyValues(pvs,filteredPds,bw.getWrappedInstance(),beanName);
if(pvs==null){
return;
}
這里通過getBeanPostProcessors方法獲得當前注冊的所有后置處理器,如果屬于InstantiationAwareBeanPostProcessor類型,則調(diào)用它的postProcessPropertyValues方法。通過遍歷,可以知道當前spring中存在7個后置處理器:
我們主要來看一下AutowiredAnnotationBeanPostProcessor,因為它負責對添加了@Autowired、@Value等注解的屬性進行依賴的填充。進入它的postProcessPropertyValues方法:
publicPropertyValuespostProcessPropertyValues(
PropertyValuespvs,PropertyDescriptor[]pds,Objectbean,StringbeanName)throwsBeanCreationException{
InjectionMetadatametadata=findAutowiringMetadata(beanName,bean.getClass(),pvs);
try{
metadata.inject(bean,beanName,pvs);
//異常處理代碼省略...
returnpvs;
}
這里的InjectionMetadata可以理解為要注入的屬性的元數(shù)據(jù),在它里面維護了一個Collection,來存放所有需要注入的bean:
privatefinalCollectionInjectedElementinjectedElements;
進入findAutowiringMetadata方法:
privateInjectionMetadatafindAutowiringMetadata(StringbeanName,Classclazz,@NullablePropertyValuespvs){
StringcacheKey=(StringUtils.hasLength(beanName)beanName:clazz.getName());
InjectionMetadatametadata=this.injectionMetadataCache.get(cacheKey);
//省略非重要代碼...
returnmetadata;
}
在執(zhí)行完這一步后,就把需要填充的屬性放進了剛才提到的injectedElements中:
接下來,繼續(xù)執(zhí)行InjectionMetadata的inject方法,在其中遍歷所有需要注入的屬性的列表,遍歷調(diào)用AutowiredAnnotationBeanPostProcessor的inject方法:
protectedvoidinject(Objectbean,@NullableStringbeanName,@NullablePropertyValuespvs)throwsThrowable{
Fieldfield=(Field)this.member;
Objectvalue;
if(this.cached){
value=resolvedCachedArgument(beanName,this.cachedFieldValue);
else{
DependencyDescriptordesc=newDependencyDescriptor(field,this.required);
desc.setContainingClass(bean.getClass());
SetStringautowiredBeanNames=newLinkedHashSet(1);
Assert.state(beanFactory!=null,"NoBeanFactoryavailable");
TypeConvertertypeConverter=beanFactory.getTypeConverter();
try{//用beanFactory解決依賴
value=beanFactory.resolveDependency(desc,beanName,autowiredBeanNames,typeConverter);
//后面代碼省略...
這里創(chuàng)建了一個DependencyDescriptor,用來維護注入屬性與它的容器類containingClass的關系,里面最重要的就是存放了注入屬性的類型、名稱,以及containingClass的類型等信息。
調(diào)用resolveDependency方法,其中沒有做什么實質(zhì)性的工作,繼續(xù)調(diào)用了doResolveDependency方法:
publicObjectdoResolveDependency(DependencyDescriptordescriptor,@NullableStringbeanName,
@NullableSetStringautowiredBeanNames,@NullableTypeConvertertypeConverter)throwsBeansException{
InjectionPointpreviousInjectionPoint=ConstructorResolver.setCurrentInjectionPoint(descriptor);
try{
Objectshortcut=descriptor.resolveShortcut(this);
if(shortcut!=null){
returnshortcut;
//依賴的屬性值的類型
Classtype=descriptor.getDependencyType();
Objectvalue=getAutowireCandidateResolver().getSuggestedValue(descriptor);
if(value!=null){
if(valueinstanceofString){
StringstrVal=resolveEmbeddedValue((String)value);
BeanDefinitionbd=(beanName!=nullcontainsBean(beanName)getMergedBeanDefinition(beanName):null);
value=evaluateBeanDefinitionString(strVal,bd);
TypeConverterconverter=(typeConverter!=nulltypeConverter:getTypeConverter());
return(descriptor.getField()!=null
converter.convertIfNecessary(value,type,descriptor.getField()):
converter.convertIfNecessary(value,type,descriptor.getMethodParameter()));
ObjectmultipleBeans=resolveMultipleBeans(descriptor,beanName,autowiredBeanNames,typeConverter);
if(multipleBeans!=null){
returnmultipleBeans;
//把匹配的值和類型拿出來,放到一個map中
MapString,ObjectmatchingBeans=findAutowireCandidates(beanName,type,descriptor);
if(matchingBeans.isEmpty()){
if(isRequired(descriptor)){
raiseNoMatchingBeanFound(type,descriptor.getResolvableType(),descriptor);
returnnull;
StringautowiredBeanName;
ObjectinstanceCandidate;
//如果有超過一個匹配的,可能會有錯誤
if(matchingBeans.size()1){
autowiredBeanName=determineAutowireCandidate(matchingBeans,descriptor);
if(autowiredBeanName==null){
if(isRequired(descriptor)||!indicatesMultipleBeans(type)){
returndescriptor.resolveNotUnique(type,matchingBeans);
else{
returnnull;
instanceCandidate=matchingBeans.get(autowiredBeanName);
else{
Map.EntryString,Objectentry=matchingBeans.entrySet().iterator().next();
autowiredBeanName=entry.getKey();
instanceCandidate=entry.getValue();
if(autowiredBeanNames!=null){
//把找到的bean的名字放到set中
autowiredBeanNames.add(autowiredBeanName);
if(instanceCandidateinstanceofClass){
//實際獲取注入的bean
instanceCandidate=descriptor.resolveCandidate(autowiredBeanName,type,this);
Objectresult=instanceCandidate;
if(resultinstanceofNullBean){
if(isRequired(descriptor)){
raiseNoMatchingBeanFound(type,descriptor.getResolvableType(),descriptor);
result=null;
if(!ClassUtils.isAssignableValue(type,result)){
thrownewBeanNotOfRequiredTypeException(autowiredBeanName,type,instanceCandidate.getClass());
returnresult;
finally{
ConstructorResolver.setCurrentInjectionPoint(previousInjectionPoint);
}
通過findAutowireCandidates方法,獲取與注入屬性匹配的值和類型,放到一個Map當中,再通過它的beanName,調(diào)用resolveCandidate方法,實際獲取注入的bean實例。這一操作底層調(diào)用的也是BeanFactory的getBean方法。
回到inject方法,使用反射將注入的bean實例賦值給屬性:
ReflectionUtils.makeAccessible(field);
field.set(bean,value);
在執(zhí)行完populateBean方法后,依賴的屬性已經(jīng)被注入成功了。
執(zhí)行回調(diào)方法及后置處理器
在bean實例化完成后,執(zhí)行各種回調(diào)和后置管理器方法:
protectedObjectinitializeBean(finalStringbeanName,finalObjectbean,@NullableRootBeanDefinitionmbd){
if(System.getSecurityManager()!=null){
AccessController.doPrivileged((PrivilegedActionObject)()-{
invokeAwareMethods(beanName,bean);
returnnull;
},getAccessControlContext());
else{
//若bean實現(xiàn)了BeanNameAware、BeanClassLoaderAware、BeanFactoryAware接口,執(zhí)行回調(diào)方法
invokeAwareMethods(beanName,bean);
ObjectwrappedBean=bean;
if(mbd==null||!mbd.isSynthetic()){
//執(zhí)行所有后置處理器的before方法
wrappedBean=applyBeanPostProcessorsBeforeInitialization(wrappedBean,beanName);
try{
//執(zhí)行bean生命周期回調(diào)中的init-method
//若bean實現(xiàn)了InitializingBean接口,執(zhí)行afterPropertiesSet方法
in
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經(jīng)權益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
- 6. 下載文件中如有侵權或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 2025浙江嘉興市海寧市海昌街道社區(qū)衛(wèi)生服務中心招聘1人備考題庫及完整答案詳解
- 2026廣西柳州市事業(yè)單位公開考試招聘工作人員1111人備考題庫附答案詳解
- 2026華中師范大學幼兒園(非事業(yè)編B類崗)招聘1人備考題庫(湖北)及完整答案詳解1套
- 2026南昌市勞動保障事務代理中心外包項目招聘人員2人備考題庫及1套完整答案詳解
- 純凈誠信購物承諾書6篇范文
- 企業(yè)數(shù)據(jù)安全使用規(guī)范承諾書范文4篇
- 中國古代文學經(jīng)典閱讀與賞析教學設計思路闡述
- XX校區(qū)2025-2026學年第一學期陽光體育活動報告
- 建設工程安全無承諾函范文8篇
- 企業(yè)營銷策劃案制作工具包
- ISO 15609-1 2019 金屬材料焊接工藝規(guī)程和評定-焊接工藝規(guī)程-電弧焊(中文版)
- 2024年四川省成都市青羊區(qū)中考數(shù)學二診試卷(含答案)
- 肥胖患者麻醉管理
- 小鯉魚跳龍門電子版
- 2019年急性腦梗死出血轉(zhuǎn)化專家共識解讀
- 左心導管檢查及造影操作技術規(guī)范
- 《混凝土結構工程施工規(guī)范》
- 社會實踐登記表
- 土地證延期申請書
- 硫乙醇酸鹽流體培養(yǎng)基適用性檢查記錄
- 進階切分技法advanced funk studies rick latham-藍色加粗字
評論
0/150
提交評論