真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

Java確保某個Bean類被最后執(zhí)行的實(shí)現(xiàn)方法-創(chuàng)新互聯(lián)

這篇文章主要介紹Java確保某個Bean類被最后執(zhí)行的實(shí)現(xiàn)方法,文中介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們一定要看完!

創(chuàng)新互聯(lián)建站主要業(yè)務(wù)有網(wǎng)站營銷策劃、成都網(wǎng)站建設(shè)、網(wǎng)站建設(shè)、微信公眾號開發(fā)、小程序設(shè)計(jì)、H5網(wǎng)站設(shè)計(jì)、程序開發(fā)等業(yè)務(wù)。一次合作終身朋友,是我們奉行的宗旨;我們不僅僅把客戶當(dāng)客戶,還把客戶視為我們的合作伙伴,在開展業(yè)務(wù)的過程中,公司還積累了豐富的行業(yè)經(jīng)驗(yàn)、成都全網(wǎng)營銷資源和合作伙伴關(guān)系資源,并逐漸建立起規(guī)范的客戶服務(wù)和保障體系。 

一、事出有因

最近有一個場景,因同一個項(xiàng)目中不同JAR包依賴同一個組件,但依賴組件的版本不同,導(dǎo)致無論使用哪個版本都報(bào)錯(無法同時兼容兩個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ì)此時有人會想說,何必這么麻煩,一個AOP切面不就搞定了嗎?通過實(shí)現(xiàn)@Around切面,把有問題的方法攔截替換成自己的適配方法邏輯,這種確實(shí)也是一種有效手段,但我認(rèn)為不夠優(yōu)雅,而且代碼的可讀性不強(qiáng)且未必能覆蓋所有方法,比如:如果涉及問題方法內(nèi)部依賴的內(nèi)部方法(如protected)過多或依賴的其它BEAN過多時,可能就會導(dǎo)致這個切面類里面要復(fù)制一堆的原問題BEAN類中的內(nèi)部方法到切面類中,但這樣帶來的風(fēng)險就是代碼重復(fù)及原代碼更新后導(dǎo)致的不一致等隱性問題,故我的原則是:如果只是簡單的替換原有方法且邏輯不復(fù)雜的可以使用AOP切面來解決,但如果涉及復(fù)雜的業(yè)務(wù)邏輯且內(nèi)部依賴過多,這時采取代理、適配或裝飾可能更為合適一些。

好了,如下就是我要分享的三種:確保一個自定義的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
@Component
public 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 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í)行時機(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)化或替換時,則可以直接采取相同的方式但指定Order在前或在后即可,也就不用這么復(fù)雜了,比如:

@Component
public class OldBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor,Order {
 
 @Override
 public int getOrder() {
 return 100;
 }
 
 ...
}

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

以上是“Java確保某個Bean類被最后執(zhí)行的實(shí)現(xiàn)方法”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關(guān)知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!


文章題目:Java確保某個Bean類被最后執(zhí)行的實(shí)現(xiàn)方法-創(chuàng)新互聯(lián)
地址分享:http://weahome.cn/article/dpcids.html

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部