1. 程式人生 > 程式設計 >Spring註解驅動擴充套件原理BeanFactoryPostProcessor

Spring註解驅動擴充套件原理BeanFactoryPostProcessor

1、擴充套件原理-BeanFactoryPostProcessor

BeanFactoryPostProcessor

* 擴充套件原理:
* BeanPostProcessor:bean後置處理器,bean建立物件初始化前後進行攔截工作的
*
* 1、BeanFactoryPostProcessor:beanFactory的後置處理器;
* 在BeanFactory標準初始化之後呼叫,來定製和修改BeanFactory的內容;
* 所有的bean定義已經儲存載入到beanFactory,但是bean的例項還未建立
*
*
* BeanFactoryPostProcessor原理:
* 1)、ioc容器建立物件

* 2)、invokeBeanFactoryPostProcessors(beanFactory);
* 如何找到所有的BeanFactoryPostProcessor並執行他們的方法;
* 1)、直接在BeanFactory中找到所有型別是BeanFactoryPostProcessor的元件,並執行他們的方法
* 2)、在初始化建立其他元件前面執行

程式碼實現 

@Component
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
 
 public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
  System.out.println("MyBeanFactoryPostProcessor...postProcessBeanFactory...");
  int count = beanFactory.getBeanDefinitionCount();
  String[] names = beanFactory.getBeanDefinitionNames();
  System.out.println("當前BeanFactory中有"+count+" 個Bean");
  System.out.println(Arrays.asList(names));
 }
 
}

2、擴充套件原理-BeanDefinitionRegistryPostProcessor

BeanDefinitionRegistryPostProcessor

* 2、BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor
* postProcessBeanDefinitionRegistry();
* 在所有bean定義資訊將要被載入,bean例項還未建立的;
*
* 優先於BeanFactoryPostProcessor執行;
* 利用BeanDefinitionRegistryPostProcessor給容器中再額外新增一些元件;

*
* 原理:
* 1)、ioc建立物件
* 2)、refresh()-》invokeBeanFactoryPostProcessors(beanFactory);
* 3)、從容器中獲取到所有的BeanDefinitionRegistryPostProcessor元件。
* 1、依次觸發所有的postProcessBeanDefinitionRegistry()方法
* 2、再來觸發postProcessBeanFactory()方法BeanFactoryPostProcessor;
*
* 4)、再來從容器中找到BeanFactoryPostProcessor元件;然後依次觸發postProcessBeanFactory()方法

程式碼實現

@Component
public class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor{
 
 public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
  // TODO Auto-generated method stub
  System.out.println("MyBeanDefinitionRegistryPostProcessor...bean的數量:"+beanFactory.getBeanDefinitionCount());
 }
 
 //BeanDefinitionRegistry Bean定義資訊的儲存中心,以後BeanFactory就是按照BeanDefinitionRegistry裡面儲存的每一個bean定義資訊建立bean例項;
 public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
  // TODO Auto-generated method stub
  System.out.println("postProcessBeanDefinitionRegistry...bean的數量:"+registry.getBeanDefinitionCount());
  //RootBeanDefinition beanDefinition = new RootBeanDefinition(Blue.class);
  AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(Blue.class).getBeanDefinition();
  registry.registerBeanDefinition("hello",beanDefinition);
 }
 
}

3、擴充套件原理-ApplicationListener用法

監聽器ApplicationListener

* 3、ApplicationListener:監聽容器中釋出的事件。事件驅動模型開發;
* public interface ApplicationListener<E extends ApplicationEvent>
* 監聽 ApplicationEvent 及其下面的子事件;
*
* 步驟:
* 1)、寫一個監聽器(ApplicationListener實現類)來監聽某個事件(ApplicationEvent及其子類)
* @EventListener;
* 原理:使用EventListenerMethodProcessor處理器來解析方法上的@EventListener;
*
* 2)、把監聽器加入到容器;
* 3)、只要容器中有相關事件的釋出,我們就能監聽到這個事件;
* ContextRefreshedEvent:容器重新整理完成(所有bean都完全建立)會發布這個事件;
* ContextClosedEvent:關閉容器會發布這個事件;
* 4)、釋出一個事件:
* applicationContext.publishEvent();

程式碼實現:

方式一:實現ApplicationListener<E extends ApplicationEvent>介面

@Component
public class MyApplicationListener implements ApplicationListener<ApplicationEvent> {
 
 //當容器中釋出此事件以後,方法觸發
 public void onApplicationEvent(ApplicationEvent event) {
  // TODO Auto-generated method stub
  System.out.println("收到事件:"+event);
 }
}

方式二:使用@EventListener註解標識事件監聽方法

@Service
public class UserService {
  
 @EventListener(classes={ApplicationEvent.class})
 public void listen(ApplicationEvent event){
  System.out.println("UserService。。監聽到的事件:"+event);
 }
}

4、擴充套件原理-ApplicationListener原理

* 原理:
* ContextRefreshedEvent、IOCTest_Ext$1[source=我釋出的時間]、ContextClosedEvent;
* 1)、ContextRefreshedEvent事件:
* 1)、容器建立物件:refresh();
* 2)、finishRefresh();容器重新整理完成會發布ContextRefreshedEvent事件
* 2)、自己釋出事件;
* 3)、容器關閉會發布ContextClosedEvent;
*
* 【事件釋出流程】:
* 3)、publishEvent(new ContextRefreshedEvent(this));
* 1)、獲取事件的多播器(派發器):getApplicationEventMulticaster()
* 2)、multicastEvent派發事件:
* 3)、獲取到所有的ApplicationListener;
* for (final ApplicationListener<?> listener : getApplicationListeners(event,type)) {
* 1)、如果有Executor,可以支援使用Executor進行非同步派發;
* Executor executor = getTaskExecutor();
* 2)、否則,同步的方式直接執行listener方法;invokeListener(listener,event);
* 拿到listener回撥onApplicationEvent方法;
*
* 【事件多播器(派發器)】
* 1)、容器建立物件:refresh();
* 2)、initApplicationEventMulticaster();初始化ApplicationEventMulticaster;
* 1)、先去容器中找有沒有id=“applicationEventMulticaster”的元件;
* 2)、如果沒有this.applicationEventMulticaster = new SimpleApplicationEventMulticaster(beanFactory);
* 並且加入到容器中,我們就可以在其他元件要派發事件,自動注入這個applicationEventMulticaster;
*
* 【容器中有哪些監聽器】
* 1)、容器建立物件:refresh();
* 2)、註冊監聽器:registerListeners();
* 從容器中拿到所有的監聽器,把他們註冊到applicationEventMulticaster中;
* String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class,true,false);
* //將listener註冊到ApplicationEventMulticaster中
* getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName);

5、擴充套件原理-@EventListener與SmartInitializingSingleton

* SmartInitializingSingleton 原理:->afterSingletonsInstantiated();
* 1)、ioc容器建立物件並refresh();
* 2)、finishBeanFactoryInitialization(beanFactory);初始化剩下的單例項bean;
* 1)、先建立所有的單例項bean;getBean();
* 2)、獲取所有建立好的單例項bean,判斷是否是SmartInitializingSingleton型別的;
* 如果是就呼叫afterSingletonsInstantiated();

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。