日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java 確保某個Bean類被最后執行的幾種實現方式

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

​ 最近有一個場景,因同一個項目中不同JAR包依賴同一個組件,但依賴組件的版本不同,導致無論使用哪個版本都報錯(無法同時兼容兩個JAR包中所需的方法調用),經過分析發現差異的部份是在一個BEAN中的方法出入參不同而矣,故考慮通過動態替換掉這個存在兼容性的BEAN,換成我們自己繼承自該BEAN類并實現適配兼容方法,從而最終解決組件版本不兼容問題;

二、解決方案困境

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

Java 確保某個Bean類被最后執行的幾種實現方式

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

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

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

第一種實現方案

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

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();}

實現方式代碼如下:

//如下是第一層自定義的BeanDefinitionRegistryPostProcessor BEAN,內部再注冊真正用于替換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()); }}

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

FirstDynamicBeanPostProcessor.postProcessBeanDefinitionRegistry (第一輪)

OldBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry (第一輪)

NewBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry (第二輪)

FirstDynamicBeanPostProcessor.postProcessBeanFactory

OldBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

NewBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

第二種實現方案

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

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);

實現方式代碼如下:

@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則將它的排序序號設置為最大 return Integer.MAX_VALUE; } return super.getOrder(obj)-1; //其余的全部設為比它小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;//確保 } }

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

FirstBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry (第1批:實現PriorityOrdered執行)

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

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

FirstBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

OldBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

NewBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

第三種實現方案

第三種:通過自定義DeferredImportSelector類并配合@Import注解,實現NewBeanDefinitionRegistryPostProcessor最后才被注冊成為BEAN,最后才有機會執行,這種方式實現思路是:因為DeferredImportSelector的執行時機是在所有@Configuration類型bean解析之后。

實現方式代碼如下:

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 { }

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

OldBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry

NewBeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry

OldBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

NewBeanDefinitionRegistryPostProcessor.postProcessBeanFactory

四、引發的思考

如上就是三種實現方式,至于哪種方式最好,這要看具體的場景,第一種、第三種影響面相對較小,而第二種因為涉及更換DependencyComparator,可能影響的是全局。另外之所以會研究如上實現方式,主要原因還是因為我們的項目框架代碼沒有考慮擴展性及規范性,比如要動態注冊BEAN,至少應實現PriorityOrdered或Order接口或指明@Order注解,這樣當我們在某些特定場景需要做一下優化或替換時,則可以直接采取相同的方式但指定Order在前或在后即可,也就不用這么復雜了,比如:

@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被最后執行的幾種實現方式的詳細內容,更多關于確保BeanDefinitionRegistryPostProcessor Bean執行的資料請關注好吧啦網其它相關文章!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
在线视频精品| 日韩精品免费一区二区在线观看| 精品国产午夜| 麻豆国产一区| 红杏一区二区三区| 国产一区二区三区不卡视频网站| 国产一区二区三区精品在线观看| 97精品在线| 亚洲成人三区| 日韩在线视频一区二区三区| 五月天激情综合网| 在线一区视频| 欧美自拍一区| 日韩国产欧美一区二区| 亚洲大全视频| 日韩一区二区三区免费视频 | 亚洲免费成人av在线| 四虎在线精品| 国产成人77亚洲精品www| 精品一区毛片| 日韩中文字幕高清在线观看| 六月婷婷一区| 国产精品三p一区二区| 久久久久美女| 欧美日韩91| 99成人在线视频| 91成人福利| 欧美亚洲激情| 国产精品theporn| 欧美成人基地| 国产日本精品| 伊人精品一区| 国产精品调教视频| 亚洲欧美不卡| av资源亚洲| 日韩欧乱色一区二区三区在线| 久久影视三级福利片| 激情综合自拍| 欧美激情综合| 中文字幕中文字幕精品| 国产精品99视频| 亚洲日韩视频| 久久精品国产大片免费观看| 国产一卡不卡| 亚洲少妇一区| 日韩不卡免费高清视频| 国产精品欧美三级在线观看| 国产农村妇女精品一区二区| 欧美丰满日韩| 国产精品v亚洲精品v日韩精品| 中文一区二区| 99久久99久久精品国产片果冰| 国产精品最新| 中文字幕中文字幕精品| 99久精品视频在线观看视频| 久久久国产精品网站| 日韩三级一区| 男人天堂欧美日韩| 91精品国产调教在线观看 | 男女男精品网站| 日韩在线观看不卡| 精品视频网站| 中文字幕av一区二区三区四区| 日韩在线精品| 精品91福利视频| 综合欧美亚洲| 模特精品在线| 免费欧美一区| 视频福利一区| 久久久亚洲欧洲日产| 亚洲日产国产精品| 国产精品毛片在线| 久久国产精品免费一区二区三区| 另类亚洲自拍| 国产亚洲一级| 亚洲婷婷免费| 欧美日韩精品在线一区| 亚洲日本网址| 欧产日产国产精品视频| 激情久久一区二区| 国产精品综合| 国产精品手机在线播放| 久久激五月天综合精品| 欧美一区91| 国产精品主播在线观看| 国产精品一国产精品k频道56| 日本一不卡视频| 天堂久久一区| 欧美久久亚洲| 久久中文字幕一区二区三区| 国产精品欧美大片| 国产a久久精品一区二区三区| 国产一区二区三区黄网站| 丝瓜av网站精品一区二区 | 伊人久久大香线蕉av超碰演员| 日韩欧美精品一区| 丝袜美腿诱惑一区二区三区 | sm久久捆绑调教精品一区| 精品久久美女| 亚洲精品永久免费视频| 国产欧美日韩亚洲一区二区三区| 日韩av在线播放中文字幕| 91成人小视频| 欧美激情视频一区二区三区免费 | 午夜国产精品视频免费体验区| 欧美一级精品| 丝袜美腿亚洲一区| 亚洲青青久久| 91p九色成人| 精品99在线| 亚洲婷婷在线| 亚洲精品大片| 精品无人区麻豆乱码久久久| 日韩久久精品网| 激情91久久| 日韩激情精品| 高清在线一区| 99综合视频| 91免费精品国偷自产在线在线| 美女视频黄 久久| 日本精品在线中文字幕| 免费精品国产| 亚洲精品在线二区| 国产精品最新自拍| 婷婷激情一区| 亚洲一区av| 免费在线欧美黄色| 欧美香蕉视频| 国产精品22p| 91免费精品| 日韩一区二区久久| 亚洲区第一页| 国产第一亚洲| 亚洲综合不卡| 国产精品大片| 波多野结衣久久精品| 视频一区二区三区入口| 国产乱人伦精品一区| 久久久蜜桃一区二区人| 日韩精品欧美精品| 日韩欧美1区| 日韩成人精品一区二区三区| 中文在线免费视频| 日韩国产成人精品| 99国产精品一区二区| 欧美日韩亚洲一区| 久久久久久久久久久9不雅视频| 日韩精品社区| 欧美精选一区二区三区| 日本午夜精品久久久久| 色天使综合视频| 国产亚洲人成a在线v网站| 亚洲91视频| 国产精品主播| 丝瓜av网站精品一区二区| 日韩1区2区| 日韩精品亚洲一区二区三区免费| 欧美亚洲日本精品| 国产精品网址| 久久亚洲精品伦理| 中文另类视频| 国产精品探花在线观看| 亚洲欧美日韩国产| 日本不卡免费高清视频在线| 综合视频一区| 国产精品88久久久久久| 欧美激情日韩| 视频一区在线播放| 欧美成人基地| 国产精品巨作av| 视频在线观看一区二区三区| 日韩中文在线播放| 国产91欧美| 国产精品一区二区三区av麻| 亚洲精品无播放器在线播放| 久久久久一区| 精品一区二区三区中文字幕| 视频在线观看一区| 亚洲先锋成人| 亚洲91精品| 欧美xxxx中国| 欧美激情三区| 久久国产三级精品| 中文字幕日本一区二区| 国产亚洲精品v| 欧美午夜不卡| 狠狠久久婷婷| 日韩欧美在线精品| 亚洲视频二区| 免费在线观看日韩欧美| 国产一区日韩欧美| 九九色在线视频| 五月天av在线| 亚洲精品一区三区三区在线观看| 日本欧美国产| 国产一区二区三区精品在线观看| 国产劲爆久久| 精品淫伦v久久水蜜桃| 精品一区二区三区中文字幕视频| 国产精品玖玖玖在线资源|