电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

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

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

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

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

二、解決方案困境

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

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

三、柳暗花明,終級(jí)解決方案

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

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

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

第一種:通過(guò)嵌套注冊(cè)自定義的BeanDefinitionRegistryPostProcessor 類(lèi)BEAN的方式,這種方式實(shí)現(xiàn)思路是:PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors會(huì)先執(zhí)行已獲得BeanDefinitionRegistryPostProcessor BEAN集合,執(zhí)行完這些BEAN集合后(這里我稱(chēng)為第一輪或第一層),會(huì)再次嘗試獲取第二輪、第三輪一直到獲取的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)部再注冊(cè)真正用于替換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()); }}//用于將原依賴(lài)的問(wèn)題Bean替換為同名的新的適配器Bean(下文中所有替換方式最終都要使用該類(lèi))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問(wèn)題Bean名稱(chēng)'); if (isContainsSpecialBean) { beanDefinitionRegistry.removeBeanDefinition('old問(wèn)題Bean名稱(chēng)'); BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(DemoCompentB.class); beanDefinitionBuilder.addConstructorArgValue(((DefaultListableBeanFactory) beanDefinitionRegistry).getBean(NewBeanAdapter.class)); //NewBeanAdapter為繼承自old問(wèn)題Bean的裝飾者、適配器類(lèi) AbstractBeanDefinition beanDefinition = beanDefinitionBuilder.getBeanDefinition(); beanDefinition.setPrimary(true); beanDefinitionRegistry.registerBeanDefinition('old問(wèn)題Bean名稱(chēng)', 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)方案

第二種:通過(guò)額外定義一個(gè)BeanDefinitionRegistryPostProcessor BEAN并實(shí)現(xiàn)PriorityOrdered、BeanFactoryAware接口,確保該BEAN最先被執(zhí)行(Order=0),然后在postProcessBeanDefinitionRegistry方法中通過(guò)applicationContext.setDependencyComparator設(shè)置自定義的排序器,達(dá)到排序BeanDefinitionRegistryPostProcessor BEAN集合的執(zhí)行順序,這種方式實(shí)現(xiàn)思路是:在執(zhí)行BeanDefinitionRegistryPostProcessor BEAN集合前會(huì)調(diào)用sortPostProcessors方法進(jìn)行排序,而排序規(guī)則又依賴(lài)于DependencyComparator,通過(guò)控制排序規(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則將它的排序序號(hào)設(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)方案

第三種:通過(guò)自定義DeferredImportSelector類(lèi)并配合@Import注解,實(shí)現(xiàn)NewBeanDefinitionRegistryPostProcessor最后才被注冊(cè)成為BEAN,最后才有機(jī)會(huì)執(zhí)行,這種方式實(shí)現(xiàn)思路是:因?yàn)镈eferredImportSelector的執(zhí)行時(shí)機(jī)是在所有@Configuration類(lèi)型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)方式,至于哪種方式最好,這要看具體的場(chǎng)景,第一種、第三種影響面相對(duì)較小,而第二種因?yàn)樯婕案鼡QDependencyComparator,可能影響的是全局。另外之所以會(huì)研究如上實(shí)現(xiàn)方式,主要原因還是因?yàn)槲覀兊捻?xiàng)目框架代碼沒(méi)有考慮擴(kuò)展性及規(guī)范性,比如要?jiǎng)討B(tài)注冊(cè)BEAN,至少應(yīng)實(shí)現(xiàn)PriorityOrdered或Order接口或指明@Order注解,這樣當(dāng)我們?cè)谀承┨囟▓?chǎ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;//只需序號(hào)在OldBeanDefinitionRegistryPostProcessor.getOrder之后即可 } ...}

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

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 香港新时代国际美容美发化妆美甲培训学校-26年培训经验,值得信赖! | 节流截止放空阀-不锈钢阀门-气动|电动截止阀-鸿华阀门有限公司 | 粉丝机械,粉丝烘干机,粉丝生产线-招远市远东粉丝机械有限公司 | 云杂志网-学术期刊-首页 | 超细粉碎机|超微气流磨|气流分级机|粉体改性设备|超微粉碎设备-山东埃尔派粉碎机厂家 | 恒温油槽-恒温水槽-低温恒温槽厂家-宁波科麦仪器有限公司 | 耐火浇注料-喷涂料-浇注料生产厂家_郑州市元领耐火材料有限公司 耐力板-PC阳光板-PC板-PC耐力板 - 嘉兴赢创实业有限公司 | 暖气片十大品牌厂家_铜铝复合暖气片厂家_暖气片什么牌子好_欣鑫达散热器 | 附着力促进剂-尼龙处理剂-PP处理剂-金属附着力处理剂-东莞市炅盛塑胶科技有限公司 | 304不锈钢无缝管_不锈钢管厂家 - 隆达钢业集团有限公司 | 药品冷藏箱厂家_低温冰箱_洁净工作台-济南欧莱博电子商务有限公司官网 | 赛尔特智能移动阳光房-阳光房厂家-赛尔特建筑科技(广东)有限公司 | 广州昊至泉水上乐园设备有限公司 | UV固化机_UVLED光固化机_UV干燥机生产厂家-上海冠顶公司专业生产UV固化机设备 | 商标转让-商标注册-商标查询-软著专利服务平台 - 赣江万网 | 氢氧化钾厂家直销批发-济南金昊化工有限公司 | 山西3A认证|太原AAA信用认证|投标AAA信用证书-山西AAA企业信用评级网 | 南京雕塑制作厂家-不锈钢雕塑制作-玻璃钢雕塑制作-先登雕塑厂 | 乐泰胶水_loctite_乐泰胶_汉高乐泰授权(中国)总代理-鑫华良供应链 | 小学教案模板_中学教师优秀教案_高中教学设计模板_教育巴巴 | 翰墨AI智能写作助手官网_人工智能问答在线AI写作免费一键生成 | 泥浆在线密度计厂家-防爆数字压力表-膜盒-远传压力表厂家-江苏大亚自控设备有限公司 | 阿里巴巴诚信通温州、台州、宁波、嘉兴授权渠道商-浙江联欣科技提供阿里会员办理 | 天坛家具官网| 高中学习网-高考生信息学习必备平台| 高中学习网-高考生信息学习必备平台| MOOG伺服阀维修,ATOS比例流量阀维修,伺服阀维修-上海纽顿液压设备有限公司 | ISO9001认证咨询_iso9001企业认证代理机构_14001|18001|16949|50430认证-艾世欧认证网 | 天一线缆邯郸有限公司_煤矿用电缆厂家_矿用光缆厂家_矿用控制电缆_矿用通信电缆-天一线缆邯郸有限公司 | 出国劳务公司_正规派遣公司[严海]| 东莞市踏板石餐饮管理有限公司_正宗桂林米粉_正宗桂林米粉加盟_桂林米粉加盟费-东莞市棒子桂林米粉 | 光环国际-新三板公司_股票代码:838504 | 山东信蓝建设有限公司官网 | 陶瓷砂磨机,盘式砂磨机,棒销式砂磨机-无锡市少宏粉体科技有限公司 | 校园气象站_超声波气象站_农业气象站_雨量监测站_风途科技 | 全温度恒温培养摇床-大容量-立式-远红外二氧化碳培养箱|南荣百科 | 合肥风管加工厂-安徽螺旋/不锈钢风管-通风管道加工厂家-安徽风之范 | 掺铥光纤放大器-C/L波段光纤放大器-小信号光纤放大器-合肥脉锐光电技术有限公司 | atcc网站,sigma试剂价格,肿瘤细胞现货,人结肠癌细胞株购买-南京科佰生物 | 冷凝水循环试验箱-冷凝水试验箱-可编程高低温试验箱厂家-上海巨为(www.juweigroup.com) | 阴离子_阳离子聚丙烯酰胺厂家_聚合氯化铝价格_水处理絮凝剂_巩义市江源净水材料有限公司 |