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

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

spring BeanProcessor接口詳解

瀏覽:24日期:2023-07-18 15:40:44
1. 簡單認識BeanProcessorBeanProcessor的理解

BeanProcessor是spring中的一個重要接口,他有兩個接口方法一個是postProcessBeforeInitialization前置初始化,另一個是postProcessAfterInitialization后置初始化。從名稱上就可以大概清楚這個接口的作用:在一個業務流程的前后加入兩個接口方法,當執行這個業務流程時,就會觸發這兩個接口方法的執行。簡單的總結一下有兩個要點:

在業務流程中,根據BeanProcessor接口方法加在不同的位置(一般是前后),可以實現對業務邏輯的擴展。 在業務邏輯執行前,BeanProcessor的實現類必須已經被創建完成(BeanProcessor接口類必須要優先實例化)。

而在spring中,就有很多實現了BeanProcessor的bean,通過在重要的業務流程(如bean的生命周期流程)的前后加上BeanProcessor接口方法,就可以對業務邏輯進行修改或補充。

一個BeanProcessor的使用實例

在spring的bean生命周期中,BeanProcessor接口方法會在bean創建后的初始化方法(init-method或@PostConstruct指向的方法)前后執行before和after方法;那有沒有在bean創建前后執行的接口方法呢?答案是肯定有的,這個功能是由BeanProcessor的子接口InstantiationAwareBeanPostProcessor來實現的,他也是有before和after方法,會在bean實例化前后執行。

spring BeanProcessor接口詳解

我們先定義一個BeanProcessor接口實現類和一個InstantiationAwareBeanPostProcessor接口實現類。

BeanPostProcessor實現類:

//net.postProcessor.CustomerPostProcessor@Componentpublic class CustomerPostProcessor implements BeanPostProcessor { @PostConstruct public void init(){ System.out.println('執行CustomerPostProcessor的PostConstruct'); } public CustomerPostProcessor(){ System.out.println('執行CustomerPostProcessor的構造方法'); } @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { System.out.println(bean+'======BeforeInitialization======'+ beanName); return bean; } @Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { System.out.println(bean+'======AfterInitialization======'+ beanName); return bean; }}

InstantiationAwareBeanPostProcessor實現類:

//net.postProcessor.CustomerInitialPostProcessor@Componentpublic class CustomerInitialPostProcessor implements InstantiationAwareBeanPostProcessor { @PostConstruct public void init(){ System.out.println('執行CustomerInitialPostProcessor的PostConstruct'); } public CustomerInitialPostProcessor(){ System.out.println('執行CustomerInitialPostProcessor的構造方法'); } @Override public Object postProcessBeforeInstantiation(Class<?> beanClass, String beanName) throws BeansException { System.out.println('bean初始化前執行:class為'+beanClass.getName()+'|beanName為'+beanName); return null; } @Override public boolean postProcessAfterInstantiation(Object bean, String beanName) throws BeansException { System.out.println('bean初始化后執行:Object為'+bean+'|beanName為'+beanName); return false; }}

再創建一個普通的bean對象:

//net.postProcessor.FirstBean@Componentpublic class FirstBean implements InitializingBean { private String msg = 'hello'; @PostConstruct public void init(){ System.out.println('執行FirstBean的PostConstruct'); } public FirstBean(){ System.out.println('FirstBean構造方法!'+msg); } public String getMsg() { return msg; } public void setMsg(String msg) { this.msg = msg; } @Override public void afterPropertiesSet() throws Exception { System.out.println('執行FirstBean的afterPropertiesSet'); }}

我們創建一個spring工廠對象將上述bean加載進去:

@Testpublic void test(){ AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext('net.postProcessor');}//執行得到以下結果:執行CustomerInitialPostProcessor的構造方法執行CustomerInitialPostProcessor的PostConstruct執行CustomerPostProcessor的構造方法執行CustomerPostProcessor的PostConstruct bean初始化前執行:class為net.postProcessor.FirstBean|beanName為firstBeanFirstBean構造方法!hellobean初始化后執行:Object為net.postProcessor.FirstBean@79179359|beanName為firstBean net.postProcessor.FirstBean@79179359======BeforeInitialization======firstBean執行FirstBean的PostConstruct執行FirstBean的afterPropertiesSetnet.postProcessor.FirstBean@79179359======AfterInitialization======firstBean

通過上述結果證明了我們之前的說法是正確的:

1.BeanPostProcessor接口類會優先實例化,且在實例化中無法不會調用BeanPostProcessor接口方法的

2.InstantiationAwareBeanPostProcessor接口方法會在FirstBean構造方法構造方法前后執行

3.BeanPostProcessor接口方法會在FirstBean實例化后進行初始化的前后執行

注意:若@PostConstruct注解方法方法未執行,請加入javax.annotation:javax.annotation-api:1.3.2jar包依賴,原因是@PostConstruct是J2EE標準的注解,不是spring自己的接口,而在JDK8往上的版本中設計者打算棄用這些注解,所以做了處理,我們是沒有辦法直接使用J2EE標準注解的(@Resource、@PostConstruct、@PreDestroy等幾個注解),為了兼容這種情況,所以有了javax.annotation-apijar包的產生(或者降低JDK版本)。

spring BeanProcessor接口詳解

2. BeanProcessor的實現思路和簡化實例BeanProcessor大概的實現思路

通過之前的了解BeanProcessor的使用,我們可以知道BeanProcessor并不復雜,但是卻十分的重要,下面來分析下BeanProcessor的實現思路:

創建個接口A,接口包含一些切點方法(Before、After、Around之類的),實現這個接口A的類要在使用前就創建好 我們需要有個業務流程,這個業務流程由若干步組成;將接口A的接口方法插入到這些業務步驟之間(需要擴展的地方) 要執行這個業務流程時,把接口A的實現類對象賦值到業務流程中,在執行業務流程中,就會觸發接口方法的執行完成功能擴展

當我們更換賦值到業務流程中的接口A的實現類時,對應的擴展邏輯也會隨之變化,這樣就實現了可插拔式的擴展邏輯(策略模式)。

一個BeanProcessor的簡化邏輯實例

在spring中我們可以創建任意數量的bean實現BeanProcessor接口,所以實際上我們是要一個全局的beanProcessorList對象用來存儲這些BeanProcessor對象;在執行業務代碼時,要循環這個beanProcessorList對象,獲取你需要的BeanProcessor對象來執行接口方法。下面是一個模擬spring bean生命周期的簡化版,來幫助你理解spring中BeanProcessor的工作原理。

net.postProcessor.SecondBean.java

@Componentpublic class SecondBean { private String msg = 'world'; public SecondBean(){ System.out.println('SecondBean構造方法!'+msg); } public String getMsg() { return msg; } public void setMsg(String msg) { this.msg = msg; }}

net.postProcessor.CustomerPostProcessor.java

@Componentpublic class CustomerPostProcessor implements BeanPostProcessor { @PostConstruct public void init(){ System.out.println('執行CustomerPostProcessor的PostConstruct'); } public CustomerPostProcessor(){ System.out.println('執行CustomerPostProcessor的構造方法'); } @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { System.out.println(bean+'======BeforeInitialization======'+ beanName); return bean; } @Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { System.out.println(bean+'======AfterInitialization======'+ beanName); return bean; }}net.postProcessor.PostProcessor.java

public class PostProcessor { //模擬掃描到的bean信息<'SecondBean', 'net.postProcessor.SecondBean'> Map<String, String> scanBeanMap = new HashMap<>(); //模擬spring的beanPostProcessors列表 List<BeanPostProcessor> processorBeanList = new ArrayList<>(); //模擬bean對象緩存 Map<String, Object> beanCache = new HashMap<>(); //添加掃描的bean信息 public PostProcessor addBeanInfo(String beanName, String classPath){ this.scanBeanMap.put(beanName, classPath); return this; } //模擬bean創建流程 public Object execute(){ try { //先臨時存儲實現了postProcessor接口的bean對象 List<BeanPostProcessor> postProcessorStrList = new ArrayList<>(); //循環scanBeanMap,獲取bean列表中實現了postProcessor接口的類,加入processorBeanList中 for(String temp: scanBeanMap.keySet()){ Class<?> clazz = Class.forName(scanBeanMap.get(temp)); //判斷是否實現了BeanPostProcessor接口 if(BeanPostProcessor.class.isAssignableFrom(clazz)){//實例化讓如臨時容器postProcessorStrList.add((BeanPostProcessor)createBean(temp)); } } //將實現了postProcessor接口的bean加入processorBeanList中 for(BeanPostProcessor obj: postProcessorStrList){ processorBeanList.add(obj); } //再次循環scanBeanMap初始化所用bean for(String temp: scanBeanMap.keySet()){ createBean(temp); } } catch (ClassNotFoundException e) { e.printStackTrace(); } return null; } //bean實例化 public Object createBean(String beanName){ //從緩存中獲取 if(beanCache.containsKey(beanName)){ return beanCache.get(beanName); }else{ //緩存中取不到,則進行創建后加入緩存 try { Class<?> clazz = Class.forName(scanBeanMap.get(beanName)); //processor前置方法執行 for(BeanPostProcessor processor : processorBeanList){processor.postProcessBeforeInitialization(clazz, beanName); } //bean實例化 Object result = clazz.getConstructor().newInstance(); //processor后置方法執行 for(BeanPostProcessor processor : processorBeanList){processor.postProcessAfterInitialization(result, beanName); } //將bean加入緩存 beanCache.put(beanName, result); return result; } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InstantiationException e) { e.printStackTrace(); } catch (NoSuchMethodException e) { e.printStackTrace(); } catch (InvocationTargetException e){ e.printStackTrace(); } } return null; }}

代碼調用

public static void main(String[] args) { PostProcessor postProcessor = new PostProcessor(); //添加掃描到的bean postProcessor .addBeanInfo('SecondBean', 'net.postProcessor.SecondBean') .addBeanInfo('CustomerPostProcessor', 'net.postProcessor.CustomerPostProcessor'); postProcessor.execute();}//執行結果執行CustomerPostProcessor的構造方法class net.postProcessor.SecondBean======BeforeInitialization======SecondBeanSecondBean構造方法!worldnet.postProcessor.SecondBean@1b40d5f0======AfterInitialization======SecondBean

代碼邏輯如下:

循環bean信息列表,將BeanPostProcessor接口bean分離出來優先實例化(實例化中緩存bean對象),并將之放入臨時容器。 循環完成,將臨時容器中的BeanPostProcessor接口bean賦值到全局BeanPostProcessor接口列表中 再次循環bean信息列表,緩存存在則直接返回緩存對象,不存在則進行bean實例化,期間循環調用全局BeanPostProcessor接口對象方法3. spring中BeanProcessor的源碼解析

我們要從spring中的refresh()開始看起:

public void refresh() throws BeansException, IllegalStateException { synchronized (this.startupShutdownMonitor) { // Prepare this context for refreshing. //刷新準備 prepareRefresh(); // Tell the subclass to refresh the internal bean factory. //告訴子類刷新內部bean工廠。 ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory(); // Prepare the bean factory for use in this context. //為容器準備bean工程 prepareBeanFactory(beanFactory); try { // Allows post-processing of the bean factory in context subclasses. //允許在上下文bean的后處理工廠子類。 postProcessBeanFactory(beanFactory); // Invoke factory processors registered as beans in the context. //優先將BeanDefinitionRegistryPostProcessorBeanFactoryPostProcessor接口的bean對象實例化 //屬于spring內部組件調用 invokeBeanFactoryPostProcessors(beanFactory); // Register bean processors that intercept bean creation. //處理用戶自定義PostProcessor接口對象,之后加入spring的beanPostProcessors列表, // 供之后預實例化其他bean時觸發這些PostProcessor方法 registerBeanPostProcessors(beanFactory);//...省略代碼 //實例化所有(non-lazy-init)單件。finishBeanFactoryInitialization(beanFactory); } catch (BeansException ex) { if (logger.isWarnEnabled()) { logger.warn('Exception encountered during context initialization - ' + 'cancelling refresh attempt: ' + ex); } // Destroy already created singletons to avoid dangling resources. //bean銷毀 destroyBeans(); // Reset ’active’ flag. //取消刷新 cancelRefresh(ex); // Propagate exception to caller. throw ex; } finally { // Reset common introspection caches in Spring’s core, since we // might not ever need metadata for singleton beans anymore... //重置公共緩存 resetCommonCaches(); } }}

其中包含有postProcess字段都有可能和BeanProcessor相關,這里有三個相關方法:

postProcessBeanFactory(beanFactory),這個是一共空的擴展方法,顯然無關 invokeBeanFactoryPostProcessors(beanFactory),處理spring中實現了BeanProcessor接口的內部組件直接調用接口方法 registerBeanPostProcessors(beanFactory),實例化用戶自定義BeanProcessor接口bean組件,之后循環賦值到全局BeanProcessor列表中

所以registerBeanPostProcessors()就是我們要找的對象,來跟進看下registerBeanPostProcessors():

//AbstractApplicationContext#registerBeanPostProcessorsprotected void registerBeanPostProcessors(ConfigurableListableBeanFactory beanFactory) { //委托給PostProcessorRegistrationDelegate.registerBeanPostProcessors進行處理PostProcessorRegistrationDelegate.registerBeanPostProcessors進行處理(beanFactory, this);}

public static void registerBeanPostProcessors( ConfigurableListableBeanFactory beanFactory, AbstractApplicationContext applicationContext) { //查詢實現了BeanPostProcessor接口的beanName String[] postProcessorNames = beanFactory.getBeanNamesForType(BeanPostProcessor.class, true, false); // Register BeanPostProcessorChecker that logs an info message when // a bean is created during BeanPostProcessor instantiation, i.e. when // a bean is not eligible for getting processed by all BeanPostProcessors. int beanProcessorTargetCount = beanFactory.getBeanPostProcessorCount() + 1 + postProcessorNames.length; beanFactory.addBeanPostProcessor(new BeanPostProcessorChecker(beanFactory, beanProcessorTargetCount)); // Separate between BeanPostProcessors that implement PriorityOrdered, // Ordered, and the rest. List<BeanPostProcessor> priorityOrderedPostProcessors = new ArrayList<>(); List<BeanPostProcessor> internalPostProcessors = new ArrayList<>(); List<String> orderedPostProcessorNames = new ArrayList<>(); List<String> nonOrderedPostProcessorNames = new ArrayList<>(); //根據beanName循環調用getBean進行實例化 for (String ppName : postProcessorNames) { if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) { BeanPostProcessor pp = beanFactory.getBean(ppName, BeanPostProcessor.class); priorityOrderedPostProcessors.add(pp); if (pp instanceof MergedBeanDefinitionPostProcessor) { internalPostProcessors.add(pp); } } else if (beanFactory.isTypeMatch(ppName, Ordered.class)) { orderedPostProcessorNames.add(ppName); } else { nonOrderedPostProcessorNames.add(ppName); } } // First, register the BeanPostProcessors that implement PriorityOrdered. //對BeanPostProcessor接口對象進行排序 sortPostProcessors(priorityOrderedPostProcessors, beanFactory); //將獲取到的PostProcessors接口對象加入到spring的beanPostProcessors列表 registerBeanPostProcessors(beanFactory, priorityOrderedPostProcessors); // Next, register the BeanPostProcessors that implement Ordered. List<BeanPostProcessor> orderedPostProcessors = new ArrayList<>(); for (String ppName : orderedPostProcessorNames) { BeanPostProcessor pp = beanFactory.getBean(ppName, BeanPostProcessor.class); orderedPostProcessors.add(pp); if (pp instanceof MergedBeanDefinitionPostProcessor) { internalPostProcessors.add(pp); } } sortPostProcessors(orderedPostProcessors, beanFactory); registerBeanPostProcessors(beanFactory, orderedPostProcessors); // Now, register all regular BeanPostProcessors. List<BeanPostProcessor> nonOrderedPostProcessors = new ArrayList<>(); for (String ppName : nonOrderedPostProcessorNames) { BeanPostProcessor pp = beanFactory.getBean(ppName, BeanPostProcessor.class); nonOrderedPostProcessors.add(pp); if (pp instanceof MergedBeanDefinitionPostProcessor) { internalPostProcessors.add(pp); } } registerBeanPostProcessors(beanFactory, nonOrderedPostProcessors); // Finally, re-register all internal BeanPostProcessors. sortPostProcessors(internalPostProcessors, beanFactory); registerBeanPostProcessors(beanFactory, internalPostProcessors); // Re-register post-processor for detecting inner beans as ApplicationListeners, // moving it to the end of the processor chain (for picking up proxies etc). beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(applicationContext));}

果然這里就是處理BeanPostProcessor接口的地方,邏輯和之前的思路類似:

循環掃描到的bean列表,獲取實現了BeanPostProcessor接口的beanName數組 循環beanName數組數組,調用beanFactory.getBean()將bean實例化,并放入priorityOrderedPostProcessors列表中 調用sortPostProcessors對priorityOrderedPostProcessors列表進行排序(處理BeanPostProcessor調用的順序) 調用registerBeanPostProcessors將priorityOrderedPostProcessors列表中的bean對象賦值到全局列表beanPostProcessors中 回到refresh()中,當調用finishBeanFactoryInitialization()對所用bean進行預實例化時就會調用這些BeanPostProcessor接口方法

以上就是spring BeanProcessor接口詳解的詳細內容,更多關于spring BeanProcessor接口的資料請關注好吧啦網其它相關文章!

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日韩在线高清| 日本亚洲欧美天堂免费| 国产欧美自拍| 国产探花一区在线观看| 欧美一区激情| 麻豆国产精品视频| 精品中国亚洲| 日本久久综合| 99久久夜色精品国产亚洲1000部| 久久精品卡一| 老牛国产精品一区的观看方式| 鲁大师成人一区二区三区| 麻豆91精品| 日本欧美在线| 麻豆国产一区| 久久精品国产网站| 精品中文字幕一区二区三区 | 天堂中文av在线资源库| 国产aⅴ精品一区二区三区久久| 成午夜精品一区二区三区软件| 92国产精品| 不卡在线一区| 亚洲精品第一| 久久丁香四色| 日韩专区精品| 先锋影音国产一区| 日韩国产一区二| 久久精品国产网站| 久久精品影视| 视频一区视频二区在线观看| 日本91福利区| 国产精品国产一区| 亚洲精品99| 欧美片网站免费| 国产精品原创| 亚洲免费激情| 国产精品亚洲四区在线观看| 亚洲黄色网址| 国产美女一区| 国产精品伦一区二区| 日韩大片在线| 中文字幕日韩欧美精品高清在线| 人人爱人人干婷婷丁香亚洲| 国产成人免费视频网站视频社区| 亚洲不卡av不卡一区二区| 丝袜国产日韩另类美女| 国产黄色精品| 99香蕉国产精品偷在线观看| 国产亚洲观看| 色爱av综合网| 亚洲aⅴ网站| 97国产精品| 亚洲精品福利| 成人羞羞在线观看网站| 日韩国产高清在线| 日韩中文影院| 91九色综合| 91av亚洲| 欧美天堂在线| 午夜av一区| 美女视频免费精品| 亚洲欧美日本国产专区一区| 久久久久久亚洲精品美女| 国产农村妇女精品一区二区| 久久不见久久见中文字幕免费| 久久亚洲成人| 久久99免费视频| 男女精品网站| 日韩在线观看| 欧美黑人做爰爽爽爽| 婷婷成人基地| 色婷婷亚洲mv天堂mv在影片| 久久福利一区| 亚洲综合在线电影| 国产美女撒尿一区二区| 亚洲综合欧美| 性欧美videohd高精| 国产欧美一区| 丝袜国产日韩另类美女| 日韩影院二区| 另类综合日韩欧美亚洲| 一级成人国产| 国产中文一区| 久久久久久自在自线| 日本欧美一区二区在线观看| 亚洲作爱视频| 四虎影视精品| 精品久久中文| 人人精品久久| 亚洲一区二区三区在线免费| 亚洲网站视频| 最新中文字幕在线播放| 国产精品网址| 奇米亚洲欧美| 亚洲精品日韩久久| 美女久久一区| 日韩视频中文| 99热精品久久| 久久久久久黄| 一区二区三区四区日本视频| 麻豆国产91在线播放| 清纯唯美亚洲综合一区| 天堂av在线一区| 黄色日韩精品| 99精品视频在线| 岛国av在线网站| 国产91在线精品| 国产精品成久久久久| 欧美黄色精品| 免费在线亚洲| 国产精品嫩模av在线| 青青青国产精品| 青青草国产成人99久久| 日韩不卡在线观看日韩不卡视频 | 蜜臀久久99精品久久久画质超高清| 日韩一区自拍| 久久视频一区| 日韩毛片在线| 欧美精品高清| 尤物tv在线精品| 1000部精品久久久久久久久| 三级小说欧洲区亚洲区| 欧美一区二区三区激情视频| 欧美影院三区| 9色国产精品| 亚洲免费一区三区| 日韩高清不卡一区二区| 欧美伊人久久| 国产精品麻豆成人av电影艾秋 | 亚洲精品一级| 日韩免费精品| 国产日产一区| 国产精品第一| 精品网站999| 亚洲国产成人二区| 九九综合九九| 手机精品视频在线观看| 日韩一区二区三区在线看| 日韩一区二区三区在线看| 日韩精品第一| 欧美aaaaaa午夜精品| 麻豆国产在线| 91精品国产福利在线观看麻豆| 欧美1区免费| 亚洲欧美久久| 91精品国产自产在线丝袜啪| 免费在线观看一区| 国产欧洲在线| 伊人久久婷婷| 天堂va在线高清一区| 国产精品密蕾丝视频下载| 欧美国产偷国产精品三区| 久久精品国产99久久| 久久国产精品亚洲77777| 日韩一区免费| 国产成人精选| 最新国产拍偷乱拍精品| 日韩国产在线一| 精品国产美女a久久9999| 成人看片网站| 亚洲久久一区| 捆绑调教美女网站视频一区 | 成人国产精品| 黄色亚洲在线| 日韩av一二三| 日韩在线二区| 亚洲狼人精品一区二区三区| 国产精品中文字幕制服诱惑| 精品伊人久久久| 亚洲精品小说| 国产日韩一区二区三区在线| 日产午夜精品一线二线三线| 9色精品在线| 国产日韩亚洲| 欧美日韩一二三四| 欧美久久精品| 99免费精品| 欧美日韩黄网站| 日韩精品一区二区三区免费观影 | 国产麻豆精品| 亚洲午夜精品久久久久久app| 老牛国产精品一区的观看方式| 国产日韩亚洲欧美精品| 91精品国产91久久久久久黑人| 日韩精品免费视频一区二区三区| 黄在线观看免费网站ktv| 美国三级日本三级久久99| 久久只有精品| 视频一区二区三区在线| 精品国产三区在线| 老鸭窝亚洲一区二区三区| 久久精品国产网站| 亚洲一区欧美| 久久精品动漫| 免费在线日韩av| 中文字幕亚洲精品乱码| 福利一区视频| 午夜精品影视国产一区在线麻豆| 日韩欧美网址| 国产精区一区二区|