關于apollo和Spring集成@Value注解通用解析
發(fā)現(xiàn)公司集成apollo后原來的@value注入的屬性不用做任何變動,也沒有換成apollo的注解,遂略看源碼后大致了解,做此筆記
1、根據(jù)文檔可知@value
是由AutowiredAnnotationBeanPostProcessor類的postProcessPropertyValues方法進行處理的,該類為InstantiationAwareBeanPostProcessor的子類
繼承關系如下圖:
2、在SpringBean創(chuàng)建三步走的第二步屬性
注入方法AbstractAutowireCapableBeanFactory.populateBean()中,可參考我上篇文章Spring啟動流程及Bean生命周期梳理。
populateBean方法中有對InstantiationAwareBeanPostProcessor接口進行調用處理,即反射獲取子類并調用方法
部分源碼如下:
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也一定是繼承類似接口進行屬性注入的
于是在Apollo中找到了SpringValueProcessor類,其繼承關系如下圖,
BeanPostProcessor的實現(xiàn)類ApolloProcessor中對postProcessAfterInitialization方法進行了實現(xiàn),該方法是Bean生命周期中最后一個執(zhí)行的方法,所以這里處理bean的屬性注入是最終的。
4、SpringValueProcessor類中
具體實現(xiàn)了Apollo對Value注解的處理,對value字段進行判斷,并注冊到自己的處理類中去處理,源碼如下
@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; } //注冊到自己的處理器 for (String key : keys) { SpringValue springValue = new SpringValue(key, value.value(), bean, beanName, field, false); springValueRegistry.register(beanFactory, key, springValue); logger.debug("Monitoring {}", springValue); } }
由此總結整體流程為:
Spring初始化時會處理BeanPostProcessor實現(xiàn)類并調用其方法,在處理到SpringValueProcessor時就會對帶有Value的屬性進行寫入,因為該方法是Bean生命周期的最后一個方法,所以是覆蓋寫入最終值
5、關于動態(tài)更新apollo會觸發(fā)更新Bean字段的原理則為
Apollo的PropertySourcesProcessor實現(xiàn)了Spring的BeanFactoryPostProcessor接口,該接口為bean生命周期最開始執(zhí)行的方法。
簡單來看有兩步。
- 第一步,初始化bean屬性
- 第二步,動態(tài)更新屬性
@Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { initializePropertySources(); initializeAutoUpdatePropertiesFeature(beanFactory); }
initializePropertySources()大致處理就是把apollo的參數(shù)加載進來放入Map中用于后面bean的屬性注入。
initializeAutoUpdatePropertiesFeature(beanFactory)方法會把beanFactory放入apollo的監(jiān)聽器中,當屬性變動時通過回調和反射來修改bean的屬性,具體源碼后續(xù)再進行分析。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
springboot+mybaties項目中掃描不到@mapper注解的解決方法
本文主要介紹了springboot+mybaties項目中掃描不到@mapper注解的解決方法,該報錯表明掃描不到Mapper層,具有一定的參考價值,感興趣的可以了解一下2024-05-05SpringMVC 上傳文件 MultipartFile 轉為 File的方法
這篇文章主要介紹了SpringMVC 上傳文件 MultipartFile 轉為 File的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-02-02springboot+jwt+springSecurity微信小程序授權登錄問題
這篇文章主要介紹了springboot+jwt+springSecurity微信小程序授權登錄問題,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-01-01