成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術(shù)文章
文章詳情頁

Java 確保某個Bean類被最后執(zhí)行的幾種實(shí)現(xiàn)方式

瀏覽:2日期:2022-08-15 15:01:43
一、事出有因

​ 最近有一個場景,因同一個項(xiàng)目中不同JAR包依賴同一個組件,但依賴組件的版本不同,導(dǎo)致無論使用哪個版本都報(bào)錯(無法同時(shí)兼容兩個JAR包中所需的方法調(diào)用),經(jīng)過分析發(fā)現(xiàn)差異的部份是在一個BEAN中的方法出入?yún)⒉煌樱士紤]通過動態(tài)替換掉這個存在兼容性的BEAN,換成我們自己繼承自該BEAN類并實(shí)現(xiàn)適配兼容方法,從而最終解決組件版本不兼容問題;

二、解決方案困境

但在實(shí)現(xiàn)的編碼過程中發(fā)現(xiàn),原依賴的那個BEAN并不是普通的通過標(biāo)注@Compent之類的注解實(shí)現(xiàn)的注冊的BEAN,而是由自定義的BeanDefinitionRegistryPostProcessor BEAN類中動態(tài)注冊的BEAN,這樣BEAN的注冊順序是“無法確定”的,我原本想通過自定義一個BeanDefinitionRegistryPostProcessor BEAN類,在postProcessBeanDefinitionRegistry方法中通過找到原依賴BEAN的名字,然后移除該名稱對應(yīng)的BEAN定義信息(BeanDefinition),最后再以原BEAN的名字定義并注冊成為我自己的適配器的BEAN類,這樣就實(shí)現(xiàn)了“移花接木”的功能,然而想法是OK的但最終運(yùn)行起來,發(fā)現(xiàn)BEAN并沒有成功被替換,究其原因發(fā)現(xiàn),原來我自己定義的BeanDefinitionRegistryPostProcessor BEAN類是優(yōu)先于原依賴的那個問題BEAN所對應(yīng)的BeanDefinitionRegistryPostProcessor BEAN類之前執(zhí)行的,這樣就會導(dǎo)致在我的自定義BeanDefinitionRegistryPostProcessor BEAN類postProcessBeanDefinitionRegistry方法中并沒有找到原依賴BEAN名字對應(yīng)的BeanDefinition,也就無法進(jìn)行正常的替換了,如果說文字難看懂,可以見如下圖所示:

Java 確保某個Bean類被最后執(zhí)行的幾種實(shí)現(xiàn)方式

三、柳暗花明,終級解決方案

既然問題根源找到,那確保一個自定義的BeanDefinitionRegistryPostProcessor 類被最后定義為Bean、且被最后執(zhí)行成為關(guān)鍵(至少得比原依賴的那個問題BEAN所對應(yīng)的BeanDefinitionRegistryPostProcessor BEAN類【如:OldBeanDefinitionRegistryPostProcessor】之后執(zhí)行才行),因?yàn)檫@樣我們才能獲得原依賴的問題Bean的BeanDefinition,才能進(jìn)行正常的替換BeanDefinition,最終達(dá)到原來依賴問題Bean的自動都依賴到新的適配器Bean,從而可以控制修改問題方法的中的邏輯(比如:兼容、降級)。當(dāng)然,我估計(jì)此時(shí)有人會想說,何必這么麻煩,一個AOP切面不就搞定了嗎?通過實(shí)現(xiàn)@Around切面,把有問題的方法攔截替換成自己的適配方法邏輯,這種確實(shí)也是一種有效手段,但我認(rèn)為不夠優(yōu)雅,而且代碼的可讀性不強(qiáng)且未必能覆蓋所有方法,比如:如果涉及問題方法內(nèi)部依賴的內(nèi)部方法(如protected)過多或依賴的其它BEAN過多時(shí),可能就會導(dǎo)致這個切面類里面要復(fù)制一堆的原問題BEAN類中的內(nèi)部方法到切面類中,但這樣帶來的風(fēng)險(xiǎn)就是代碼重復(fù)及原代碼更新后導(dǎo)致的不一致等隱性問題,故我的原則是:如果只是簡單的替換原有方法且邏輯不復(fù)雜的可以使用AOP切面來解決,但如果涉及復(fù)雜的業(yè)務(wù)邏輯且內(nèi)部依賴過多,這時(shí)采取代理、適配或裝飾可能更為合適一些。

好了,如下就是我要分享的三種:確保一個自定義的BeanDefinitionRegistryPostProcessor 類被最后定義為Bean、且被最后執(zhí)行的實(shí)現(xiàn)方式。

第一種實(shí)現(xiàn)方案

第一種:通過嵌套注冊自定義的BeanDefinitionRegistryPostProcessor 類BEAN的方式,這種方式實(shí)現(xiàn)思路是:PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors會先執(zhí)行已獲得BeanDefinitionRegistryPostProcessor BEAN集合,執(zhí)行完這些BEAN集合后(這里我稱為第一輪或第一層),會再次嘗試獲取第二輪、第三輪一直到獲取的BeanDefinitionRegistryPostProcessor BEAN集合全部處理完成為止,框架相關(guān)代碼片段如下:

boolean reiterate = true;while (reiterate) {reiterate = false;postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);for (String ppName : postProcessorNames) {if (!processedBeans.contains(ppName)) {currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));processedBeans.add(ppName);reiterate = true;}}sortPostProcessors(currentRegistryProcessors, beanFactory);registryProcessors.addAll(currentRegistryProcessors);invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);currentRegistryProcessors.clear();}

實(shí)現(xiàn)方式代碼如下:

//如下是第一層自定義的BeanDefinitionRegistryPostProcessor BEAN,內(nèi)部再注冊真正用于替換BEAN目的NewBeanDefinitionRegistryPostProcessor BEAN//author:zuowenjun@Componentpublic class FirstDynamicBeanPostProcessor implements BeanDefinitionRegistryPostProcessor { @Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry beanDefinitionRegistry) throws BeansException { BeanDefinitionBuilder beanDefinitionBuilder=BeanDefinitionBuilder.genericBeanDefinition(NewBeanDefinitionRegistryPostProcessor.class);beanDefinitionRegistry.registerBeanDefinition('newBeanDefinitionRegistryPostProcessor',beanDefinitionBuilder.getBeanDefinition()); System.out.printf('【%1$tF %1$tT.%1$tL】%s,FirstDynamicBeanPostProcessor.postProcessBeanDefinitionRegistry%n', new Date()); } @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory configurableListableBeanFactory) throws BeansException { System.out.printf('【%1$tF %1$tT.%1$tL】%s,FirstDynamicBeanPostProcessor.postProcessBeanFactory%n', new Date()); }}//用于將原依賴的問題Bean替換為同名的新的適配器Bean(下文中所有替換方式最終都要使用該類)public class NewBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor { @Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry beanDefinitionRegistry) throws BeansException { System.out.printf('【%1$tF %1$tT.%1$tL】%s,NewBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry%n',new Date()); boolean isContainsSpecialBean = ((DefaultListableBeanFactory) beanDefinitionRegistry).containsBean('old問題Bean名稱'); if (isContainsSpecialBean) { beanDefinitionRegistry.removeBeanDefinition('old問題Bean名稱'); BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(DemoCompentB.class); beanDefinitionBuilder.addConstructorArgValue(((DefaultListableBeanFactory) beanDefinitionRegistry).getBean(NewBeanAdapter.class)); //NewBeanAdapter為繼承自old問題Bean的裝飾者、適配器類 AbstractBeanDefinition beanDefinition = beanDefinitionBuilder.getBeanDefinition(); beanDefinition.setPrimary(true); beanDefinitionRegistry.registerBeanDefinition('old問題Bean名稱', beanDefinition); } } @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory configurableListableBeanFactory) throws BeansException { System.out.printf('【%1$tF %1$tT.%1$tL】%s,NewBeanDefinitionRegistryPostProcessor.postProcessBeanFactory%n',new Date()); }}

最終執(zhí)行的順序如下:(可以看到NewBeanDefinitionRegistryPostProcessor是在OldBeanDefinitionRegistryPostProcessor之后執(zhí)行的,這樣就可以正常替換Bean定義了)

FirstDynamicBeanPostProcessor.postProcessBeanDefinitionRegistry (第一輪)

OldBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry (第一輪)

NewBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry (第二輪)

FirstDynamicBeanPostProcessor.postProcessBeanFactory

OldBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

NewBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

第二種實(shí)現(xiàn)方案

第二種:通過額外定義一個BeanDefinitionRegistryPostProcessor BEAN并實(shí)現(xiàn)PriorityOrdered、BeanFactoryAware接口,確保該BEAN最先被執(zhí)行(Order=0),然后在postProcessBeanDefinitionRegistry方法中通過applicationContext.setDependencyComparator設(shè)置自定義的排序器,達(dá)到排序BeanDefinitionRegistryPostProcessor BEAN集合的執(zhí)行順序,這種方式實(shí)現(xiàn)思路是:在執(zhí)行BeanDefinitionRegistryPostProcessor BEAN集合前會調(diào)用sortPostProcessors方法進(jìn)行排序,而排序規(guī)則又依賴于DependencyComparator,通過控制排序規(guī)則實(shí)現(xiàn)間接控制執(zhí)行順序,先看框架的代碼片段:

private static void sortPostProcessors(List<?> postProcessors, ConfigurableListableBeanFactory beanFactory) {Comparator<Object> comparatorToUse = null;if (beanFactory instanceof DefaultListableBeanFactory) {comparatorToUse = ((DefaultListableBeanFactory) beanFactory).getDependencyComparator();}if (comparatorToUse == null) {comparatorToUse = OrderComparator.INSTANCE;}postProcessors.sort(comparatorToUse);}//如下是invokeBeanFactoryPostProcessors方法片段:sortPostProcessors(currentRegistryProcessors, beanFactory);registryProcessors.addAll(currentRegistryProcessors);invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);

實(shí)現(xiàn)方式代碼如下:

@Component public static class FirstBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor, PriorityOrdered , BeanFactoryAware { private BeanFactory beanFactory; @Override public void setBeanFactory(BeanFactory beanFactory) throws BeansException { this.beanFactory=beanFactory; } @Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry beanDefinitionRegistry) throws BeansException { ((DefaultListableBeanFactory) beanFactory).setDependencyComparator(new OrderComparator(){ @Override protected int getOrder(Object obj) { if (obj instanceof NewBeanDefinitionRegistryPostProcessor){ //如果是NewBeanDefinitionRegistryPostProcessor則將它的排序序號設(shè)置為最大 return Integer.MAX_VALUE; } return super.getOrder(obj)-1; //其余的全部設(shè)為比它小1 } }); System.out.printf('【%1$tF %1$tT.%1$tL】%s,FirstBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry%n', new Date()); } @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory configurableListableBeanFactory) throws BeansException { System.out.printf('【%1$tF %1$tT.%1$tL】%s,FirstBeanDefinitionRegistryPostProcessor.postProcessBeanFactory%n', new Date()); } @Override public int getOrder() { return 0;//確保 } }

最終執(zhí)行的順序如下:(NewBeanDefinitionRegistryPostProcessor在OldBeanDefinitionRegistryPostProcessor后面執(zhí)行)

FirstBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry (第1批:實(shí)現(xiàn)PriorityOrdered執(zhí)行)

OldBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry (第3批:普通BEAN執(zhí)行)

NewBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry (第3批:普通BEAN執(zhí)行)

FirstBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

OldBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

NewBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

第三種實(shí)現(xiàn)方案

第三種:通過自定義DeferredImportSelector類并配合@Import注解,實(shí)現(xiàn)NewBeanDefinitionRegistryPostProcessor最后才被注冊成為BEAN,最后才有機(jī)會執(zhí)行,這種方式實(shí)現(xiàn)思路是:因?yàn)镈eferredImportSelector的執(zhí)行時(shí)機(jī)是在所有@Configuration類型bean解析之后。

實(shí)現(xiàn)方式代碼如下:

public static class BeansImportSelector implements DeferredImportSelector { @Override public String[] selectImports(AnnotationMetadata importingClassMetadata) { return new String[]{NewBeanDefinitionRegistryPostProcessor.class.getName()}; } }@Configuration@Import(BeansImportSelector.class)public class BeansConfig { }

最終執(zhí)行的順序如下:(NewBeanDefinitionRegistryPostProcessor在OldBeanDefinitionRegistryPostProcessor后面執(zhí)行)

OldBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry

NewBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry

OldBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

NewBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

四、引發(fā)的思考

如上就是三種實(shí)現(xiàn)方式,至于哪種方式最好,這要看具體的場景,第一種、第三種影響面相對較小,而第二種因?yàn)樯婕案鼡QDependencyComparator,可能影響的是全局。另外之所以會研究如上實(shí)現(xiàn)方式,主要原因還是因?yàn)槲覀兊捻?xiàng)目框架代碼沒有考慮擴(kuò)展性及規(guī)范性,比如要動態(tài)注冊BEAN,至少應(yīng)實(shí)現(xiàn)PriorityOrdered或Order接口或指明@Order注解,這樣當(dāng)我們在某些特定場景需要做一下優(yōu)化或替換時(shí),則可以直接采取相同的方式但指定Order在前或在后即可,也就不用這么復(fù)雜了,比如:

@Componentpublic class OldBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor,Order { @Override public int getOrder() { return 100; } ...}@Componentpublic class NewBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor,Order { @Override public int getOrder() { return 101;//只需序號在OldBeanDefinitionRegistryPostProcessor.getOrder之后即可 } ...}

以上就是Java 確保某個BeanDefinitionRegistryPostProcessor Bean被最后執(zhí)行的幾種實(shí)現(xiàn)方式的詳細(xì)內(nèi)容,更多關(guān)于確保BeanDefinitionRegistryPostProcessor Bean執(zhí)行的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Java
相關(guān)文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
久久亚洲一区二区三区明星换脸 | 欧美日韩免费视频| 久久久久亚洲综合| 精品在线一区二区三区| 香蕉精品999视频一区二区| 国产精品美女久久久久久久| 国产不卡在线一区| 欧美日韩极品在线观看一区| 蜜臀久久99精品久久久久久9| 99www免费人成精品| 中文字幕av一区二区三区免费看| 91福利在线免费观看| 成人在线视频首页| 国产精品动漫网站| 亚洲精品一区二区三区福利 | 美腿丝袜亚洲综合| 亚洲精品久久| 亚洲国产电影在线观看| 欧洲生活片亚洲生活在线观看| 成人福利视频网站| 欧美日韩免费观看一区三区| 亚洲国产精品影院| **欧美大码日韩| 午夜日韩激情| 国产午夜精品久久| 欧美成人有码| 日本一区二区视频在线观看| 94-欧美-setu| 欧美激情在线免费观看| 成人午夜视频在线观看| 日韩欧美成人一区| av在线播放不卡| 久久这里只有精品首页| 成人黄色777网| 日韩免费在线观看| 成人91在线观看| 国产婷婷色一区二区三区在线| 你懂的视频一区二区| 国产欧美日韩精品一区| 欧美午夜精品理论片a级大开眼界 欧美午夜精品久久久久免费视 | 亚洲综合清纯丝袜自拍| 午夜一区不卡| 久久精品国产亚洲一区二区三区 | 最新中文字幕一区二区三区| 亚洲国产高清一区二区三区| 亚洲激情五月婷婷| 噜噜噜91成人网| 麻豆精品一区二区| 91精品国产入口| eeuss鲁片一区二区三区| 国产亚洲综合性久久久影院| 国产精品久久| 亚洲激情五月婷婷| 色香蕉久久蜜桃| 精品夜夜嗨av一区二区三区| 欧美日韩国产影片| 国产91精品一区二区麻豆亚洲| 26uuu国产日韩综合| 国产在线精品二区| 一区二区久久久久| 欧美性猛片aaaaaaa做受| 国产成人高清视频| 国产情人综合久久777777| 国自产拍偷拍福利精品免费一 | 日本欧美肥老太交大片| 欧美二区乱c少妇| av网站一区二区三区| 国产欧美在线观看一区| 伊人色综合久久天天五月婷| 亚洲国产精品自拍| 欧美日韩视频一区二区| 国产91富婆露脸刺激对白| 国产拍揄自揄精品视频麻豆| 另类国产ts人妖高潮视频| 精品影视av免费| 亚洲国产精品成人综合色在线婷婷| 国产三级精品在线不卡| 久久99久久99精品免视看婷婷| 精品国产1区二区| 99精品国产高清一区二区| 免费一级片91| 精品国产1区2区3区| 亚洲欧洲日本mm| 久久精品国产澳门| 欧美sm美女调教| 国产日韩在线一区二区三区| 国产在线视视频有精品| 国产精品国产三级国产三级人妇 | 国产午夜精品一区二区三区视频 | 国产亚洲va综合人人澡精品| 亚洲欧美日本视频在线观看| 国产成人久久精品77777最新版本| 国产精品视频一区二区三区不卡| 久久久久久久久久久久久久一区 | 99视频精品全部免费在线| 亚洲欧美激情插| 欧美军同video69gay| 国内精品99| 日本aⅴ亚洲精品中文乱码| 精品精品国产高清a毛片牛牛 | 亚洲免费av高清| 欧美日韩国产另类一区| 在线观看亚洲| 激情综合网最新| 亚洲色图第一区| 日韩欧美成人激情| 色琪琪一区二区三区亚洲区| 欧美日韩视频一区二区三区| 另类小说图片综合网| 国产精品久久久久久久久动漫| 欧美日韩免费视频| 国产亚洲精品久久飘花| 成人听书哪个软件好| 日韩福利电影在线观看| 久久久www免费人成精品| 久久精品女人| 91在线看国产| 麻豆精品国产传媒mv男同| 一区精品在线播放| 日韩三级.com| 久久精品亚洲| eeuss国产一区二区三区| 日韩1区2区3区| 亚洲欧美一区二区在线观看| 91精品国产免费| 色婷婷久久综合| 亚洲久久视频| 成人h版在线观看| 日韩激情视频网站| 国产精品沙发午睡系列990531| 欧美美女激情18p| 亚久久调教视频| 在线观看欧美一区| 97久久超碰国产精品电影| 精品一区二区三区的国产在线播放 | 成人性色生活片免费看爆迷你毛片| 五月综合激情婷婷六月色窝| 中文字幕日本乱码精品影院| 精品999久久久| 777xxx欧美| 日本精品视频一区二区| 亚洲一区二区精品在线| 韩日在线一区| 91在线国产福利| 国产成人av电影免费在线观看| 日韩av网站免费在线| 亚洲综合在线免费观看| 国产婷婷精品av在线| 欧美日韩情趣电影| 久久久久综合| 一区二区激情| 狠狠色噜噜狠狠色综合久| aaa亚洲精品| 国产福利不卡视频| 久久精品99国产精品| 五月天精品一区二区三区| 亚洲免费毛片网站| 国产日韩欧美亚洲| 欧美xxxx老人做受| 欧美一级久久久| 亚洲一二区在线| 国产成人精品影视| 久久疯狂做爰流白浆xx| 亚洲v中文字幕| 亚洲在线视频一区| 亚洲女与黑人做爰| 日本一区二区三区高清不卡| 精品久久久久久久久久久久包黑料 | 一区二区三区电影在线播| 中文字幕免费不卡在线| 久久久久久亚洲综合影院红桃 | 洋洋av久久久久久久一区| 亚洲色图一区二区三区| 中文一区一区三区高中清不卡| 久久精品男人天堂av| 久久久久久久久99精品| 亚洲精品一区二区三区影院| 欧美成人猛片aaaaaaa| 欧美一级高清片| 日韩一区二区在线播放| 欧美久久久久久久久久| 欧美精品精品一区| 69堂国产成人免费视频| 欧美日韩精品一区二区三区| 欧美日韩一区中文字幕| 欧美在线一二三| 欧美综合在线视频| 欧美日韩精品欧美日韩精品| 欧美三区免费完整视频在线观看| 欧美在线制服丝袜| 91精品国产综合久久精品app| 日韩三级高清在线| 久久综合精品国产一区二区三区| 欧美不卡一二三| 久久综合一区二区| 国产日韩欧美电影| 亚洲欧美中日韩| 一区二区三区在线观看动漫| 午夜精品一区二区三区三上悠亚| 丝袜美腿亚洲色图|