關(guān)于apollo和Spring集成@Value注解通用解析
發(fā)現(xiàn)公司集成apollo后原來的@value注入的屬性不用做任何變動(dòng),也沒有換成apollo的注解,遂略看源碼后大致了解,做此筆記
1、根據(jù)文檔可知@value
是由AutowiredAnnotationBeanPostProcessor類的postProcessPropertyValues方法進(jìn)行處理的,該類為InstantiationAwareBeanPostProcessor的子類
繼承關(guān)系如下圖:

2、在SpringBean創(chuàng)建三步走的第二步屬性
注入方法AbstractAutowireCapableBeanFactory.populateBean()中,可參考我上篇文章Spring啟動(dòng)流程及Bean生命周期梳理。
populateBean方法中有對(duì)InstantiationAwareBeanPostProcessor接口進(jìn)行調(diào)用處理,即反射獲取子類并調(diào)用方法
部分源碼如下:
if (hasInstAwareBpps) {
if (pvs == null) {
pvs = mbd.getPropertyValues();
}
for (BeanPostProcessor bp : getBeanPostProcessors()) {
if (bp instanceof InstantiationAwareBeanPostProcessor) {
InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;
PropertyValues pvsToUse = ibp.postProcessProperties(pvs, bw.getWrappedInstance(), beanName);
if (pvsToUse == null) {
if (filteredPds == null) {
filteredPds = filterPropertyDescriptorsForDependencyCheck(bw, mbd.allowCaching);
}
pvsToUse = ibp.postProcessPropertyValues(pvs, filteredPds, bw.getWrappedInstance(), beanName);
if (pvsToUse == null) {
return;
}
}
pvs = pvsToUse;
}
}
}
3、想必Appllo也一定是繼承類似接口進(jìn)行屬性注入的
于是在Apollo中找到了SpringValueProcessor類,其繼承關(guān)系如下圖,

BeanPostProcessor的實(shí)現(xiàn)類ApolloProcessor中對(duì)postProcessAfterInitialization方法進(jìn)行了實(shí)現(xiàn),該方法是Bean生命周期中最后一個(gè)執(zhí)行的方法,所以這里處理bean的屬性注入是最終的。
4、SpringValueProcessor類中
具體實(shí)現(xiàn)了Apollo對(duì)Value注解的處理,對(duì)value字段進(jìn)行判斷,并注冊(cè)到自己的處理類中去處理,源碼如下
@Override
protected void processField(Object bean, String beanName, Field field) {
// register @Value on field
Value value = field.getAnnotation(Value.class);
if (value == null) {
return;
}
//解析value中自定義的屬性
Set<String> keys = placeholderHelper.extractPlaceholderKeys(value.value());
if (keys.isEmpty()) {
return;
}
//注冊(cè)到自己的處理器
for (String key : keys) {
SpringValue springValue = new SpringValue(key, value.value(), bean, beanName, field, false);
springValueRegistry.register(beanFactory, key, springValue);
logger.debug("Monitoring {}", springValue);
}
}
由此總結(jié)整體流程為:
Spring初始化時(shí)會(huì)處理BeanPostProcessor實(shí)現(xiàn)類并調(diào)用其方法,在處理到SpringValueProcessor時(shí)就會(huì)對(duì)帶有Value的屬性進(jìn)行寫入,因?yàn)樵摲椒ㄊ荁ean生命周期的最后一個(gè)方法,所以是覆蓋寫入最終值
5、關(guān)于動(dòng)態(tài)更新apollo會(huì)觸發(fā)更新Bean字段的原理則為
Apollo的PropertySourcesProcessor實(shí)現(xiàn)了Spring的BeanFactoryPostProcessor接口,該接口為bean生命周期最開始執(zhí)行的方法。
簡(jiǎn)單來看有兩步。
- 第一步,初始化bean屬性
- 第二步,動(dòng)態(tài)更新屬性
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
initializePropertySources();
initializeAutoUpdatePropertiesFeature(beanFactory);
}
initializePropertySources()大致處理就是把a(bǔ)pollo的參數(shù)加載進(jìn)來放入Map中用于后面bean的屬性注入。
initializeAutoUpdatePropertiesFeature(beanFactory)方法會(huì)把beanFactory放入apollo的監(jiān)聽器中,當(dāng)屬性變動(dòng)時(shí)通過回調(diào)和反射來修改bean的屬性,具體源碼后續(xù)再進(jìn)行分析。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java整合mybatis實(shí)現(xiàn)過濾數(shù)據(jù)
這篇文章主要介紹了Java整合mybatis實(shí)現(xiàn)過濾數(shù)據(jù),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2023-01-01
SpringBoot 如何實(shí)現(xiàn)自定義Redis序列化
這篇文章主要介紹了SpringBoot 如何實(shí)現(xiàn)自定義Redis序列化方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
詳解如何為SpringBoot Web應(yīng)用的日志方便追蹤
在Web應(yīng)用程序領(lǐng)域,有效的請(qǐng)求監(jiān)控和可追溯性對(duì)于維護(hù)系統(tǒng)完整性和診斷問題至關(guān)重要,SpringBoot是一種用于構(gòu)建Java應(yīng)用程序的流行框架,在本文中,我們探討了在SpringBoot中向日志添加唯一ID的重要性,需要的朋友可以參考下2023-11-11
基于Java驗(yàn)證jwt token代碼實(shí)例
這篇文章主要介紹了基于Java驗(yàn)證jwt token代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
淺談Spring中@Transactional事務(wù)回滾及示例(附源碼)
本篇文章主要介紹了淺談Spring中@Transactional事務(wù)回滾及示例(附源碼),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-12-12
Java設(shè)計(jì)模塊系列之書店管理系統(tǒng)單機(jī)版(二)
這篇文章主要為大家詳細(xì)介紹了Java單機(jī)版的書店管理系統(tǒng)設(shè)計(jì)模塊和思想第二章,感興趣的小伙伴們可以參考一下2016-08-08

