有時所在項目忙于業(yè)務(wù)快速迭代,只知道如何應(yīng)用框架組件橡娄,底層原理卻忽略了诗箍,框架許多精妙設(shè)計沒掌握到。每天重復(fù)流水線工作瀑踢,業(yè)務(wù)層面有所提升扳还,但技術(shù)原地不動,每次熱情高漲探索底層實現(xiàn)橱夭,由于種種原因,堅持沒多久就放棄了桑逝。為了解開放棄探索小伙伴們心中疑惑棘劣,本人花了一些時間研究了底層實現(xiàn)。此次分享學習心得為Spring 依賴注入原理楞遏,在實際項目中最常見的依賴注入場景茬暇,跨業(yè)務(wù)注解服務(wù)互相調(diào)用首昔,配置文件引用第三方服務(wù)時信息配置。
一 什么是依賴注入
通過類setXxx函數(shù)糙俗,有參構(gòu)造函數(shù)或注解形式為類的成員屬性附上值
二 依賴注入有以下方法
下面這幾張注入方式經(jīng)常碰到勒奇,在工作一段時間后,沉下心翻看源碼后巧骚,多年的疑惑才解開赊颠。
1 通過xml配置文件為類實例屬性附上值常用有
1.1 bean子標簽<property name="name" value="test"/> 標簽配上值,property標簽的name屬性值在類中需要對應(yīng)public權(quán)限setXxx函數(shù)如下
XML文件配置
<bean id="dataSource" class="test.Db">
<property name="name" value="QQ"/>
</bean>
JAVA類構(gòu)建
package test;
public class Db {
private String name;
public void setName(String name) {
this.name = name;
}
}
1.2 bean子標簽<constructor-arg name="name" value="test"/> 標簽配上值,constructor-arg標簽的name屬性值劈彪,在類中構(gòu)造函數(shù)需要對應(yīng)參數(shù)如下
XML文件配置
<bean id="dataSource" class="test.Db" >
? ? ? ? <constructor-arg name="name" value="qq"/>
</bean>
JAVA類構(gòu)建
public class Db {
private String name;
public Db(String name) {
this.name = name;
}
}
2 類屬性加上屬性自動注入注解
package test;
public class Db {
@Value("${name}")
private String name;
}
package test;
public class Db {
@Autowired
private X x;
}
package test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.stereotype.Service;
@Service
public class X {
}
三 依賴注入原理
上面的幾種形式在實際項目經(jīng)常碰到竣蹦,如何實現(xiàn)呢?
1 基于bean子標簽property注入原理
1.1 加載bean配置文件沧奴,解析bean標簽屬性值痘括,賦值給BeanDefinition實例并存進集合中,執(zhí)行
finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory)滔吠,實例化bean(使用反射技術(shù)實例化)纲菌。
再調(diào)用populateBean(String beanName, RootBeanDefinition mbd, @Nullable BeanWrapper bw),內(nèi)部執(zhí)行使用反射技術(shù)(
ReflectionUtils.makeAccessible(writeMethod);
writeMethod.invoke(getWrappedInstance(), value); writeMethod為Method實例)疮绷,最后調(diào)用函數(shù)setXxx為屬性賦值驰后。
2 基于bean子標簽constructor-arg注入原理
2.1 加載bean配置文件,解析bean標簽屬性值矗愧,賦值給BeanDefinition實例并存進集合中灶芝,執(zhí)行
finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory),當bean的factoryMethodName屬性值為空唉韭,bean的有參構(gòu)造函數(shù)多個且調(diào)用了夜涕。執(zhí)行函數(shù)autowireConstructor(String beanName, RootBeanDefinition mbd, @Nullable Constructor<?>[] ctors, @Nullable Object[] explicitArgs)傳入?yún)?shù)。
通過BeanDefinition的成員屬性constructorArgument獲取構(gòu)造函數(shù)參數(shù)属愤,
再執(zhí)行函數(shù)
resolveConstructorArguments(String beanName, RootBeanDefinition mbd, BeanWrapper bw,ConstructorArgumentValues cargs, ConstructorArgumentValues resolvedValues)傳入?yún)?shù)女器,循環(huán)遍歷構(gòu)造函數(shù)實例,獲取出匹配的構(gòu)造函數(shù)住诸。
通過反射實例化并通過構(gòu)造函數(shù)參數(shù)為類屬性賦值(即執(zhí)行ctor.newInstance(argsWithDefaultValues)驾胆,ctor為Constructor<T>類型)。
3 基于注解注入待續(xù)