有時(shí)所在項(xiàng)目忙于業(yè)務(wù)快速迭代,只知道如何應(yīng)用框架組件,底層原理卻忽略了,框架許多精妙設(shè)計(jì)沒掌握到。每天重復(fù)流水線工作,業(yè)務(wù)層面有所提升,但技術(shù)原地不動(dòng),每次熱情高漲探索底層實(shí)現(xiàn),由于種種原因,堅(jiān)持沒多久就放棄了。為了解開放棄探索小伙伴們心中疑惑,本人花了一些時(shí)間研究了底層實(shí)現(xiàn)。此次分享學(xué)習(xí)心得為Spring 依賴注入原理,在實(shí)際項(xiàng)目中最常見的依賴注入場景,跨業(yè)務(wù)注解服務(wù)互相調(diào)用,配置文件引用第三方服務(wù)時(shí)信息配置。
一 什么是依賴注入
通過類setXxx函數(shù),有參構(gòu)造函數(shù)或注解形式為類的成員屬性附上值
二 依賴注入有以下方法
下面這幾張注入方式經(jīng)常碰到,在工作一段時(shí)間后,沉下心翻看源碼后,多年的疑惑才解開。
1 通過xml配置文件為類實(shí)例屬性附上值常用有
1.1 bean子標(biāo)簽<property name="name" value="test"/> 標(biāo)簽配上值,property標(biāo)簽的name屬性值在類中需要對(duì)應(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子標(biāo)簽<constructor-arg name="name" value="test"/> 標(biāo)簽配上值,constructor-arg標(biāo)簽的name屬性值,在類中構(gòu)造函數(shù)需要對(duì)應(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 類屬性加上屬性自動(dòng)注入注解
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 {
}
三 依賴注入原理
上面的幾種形式在實(shí)際項(xiàng)目經(jīng)常碰到,如何實(shí)現(xiàn)呢?
1 基于bean子標(biāo)簽property注入原理
1.1 加載bean配置文件,解析bean標(biāo)簽屬性值,賦值給BeanDefinition實(shí)例并存進(jìn)集合中,執(zhí)行
finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory),實(shí)例化bean(使用反射技術(shù)實(shí)例化)。
再調(diào)用populateBean(String beanName, RootBeanDefinition mbd, @Nullable BeanWrapper bw),內(nèi)部執(zhí)行使用反射技術(shù)(
ReflectionUtils.makeAccessible(writeMethod);
writeMethod.invoke(getWrappedInstance(), value); writeMethod為Method實(shí)例),最后調(diào)用函數(shù)setXxx為屬性賦值。
2 基于bean子標(biāo)簽constructor-arg注入原理
2.1 加載bean配置文件,解析bean標(biāo)簽屬性值,賦值給BeanDefinition實(shí)例并存進(jìn)集合中,執(zhí)行
finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory),當(dāng)bean的factoryMethodName屬性值為空,bean的有參構(gòu)造函數(shù)多個(gè)且調(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ù)實(shí)例,獲取出匹配的構(gòu)造函數(shù)。
通過反射實(shí)例化并通過構(gòu)造函數(shù)參數(shù)為類屬性賦值(即執(zhí)行ctor.newInstance(argsWithDefaultValues),ctor為Constructor<T>類型)。
3 基于注解注入待續(xù)