spring-atguigu-其他的属性---39-43

/**
 * 扩展原理:
 * BeanPostProcessor:bean后置处理器,bean创建对象初始化前后进行拦截工作的
 * 
 * 1、BeanFactoryPostProcessor:beanFactory的后置处理器;
 * 		在BeanFactory标准初始化之后调用,来定制和修改BeanFactory的内容;
 * 		所有的bean定义已经保存加载到beanFactory,但是bean的实例还未创建
 * 
 * 
 * BeanFactoryPostProcessor原理:
 * 1)、ioc容器创建对象
 * 2)、invokeBeanFactoryPostProcessors(beanFactory);
 * 		如何找到所有的BeanFactoryPostProcessor并执行他们的方法;
 * 			1)、直接在BeanFactory中找到所有类型是BeanFactoryPostProcessor的组件,并执行他们的方法
 * 			2)、在初始化创建其他组件前面执行
 * 
 * 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()方法
 * 	
 * 3、ApplicationListener:监听容器中发布的事件。事件驱动模型开发;
 * 	  public interface ApplicationListener<E extends ApplicationEvent>
 * 		监听 ApplicationEvent 及其下面的子事件;
 * 
 * 	 步骤:
 * 		1)、写一个监听器(ApplicationListener实现类)来监听某个事件(ApplicationEvent及其子类)
 * 			@EventListener;
 * 			原理:使用EventListenerMethodProcessor处理器来解析方法上的@EventListener;
 * 
 * 		2)、把监听器加入到容器;
 * 		3)、只要容器中有相关事件的发布,我们就能监听到这个事件;
 * 				ContextRefreshedEvent:容器刷新完成(所有bean都完全创建)会发布这个事件;
 * 				ContextClosedEvent:关闭容器会发布这个事件;
 * 		4)、发布一个事件:
 * 				applicationContext.publishEvent();
 * 	
 *  原理:
 *  	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);
 *  		
 *   SmartInitializingSingleton 原理:->afterSingletonsInstantiated();
 *   		1)、ioc容器创建对象并refresh();
 *   		2)、finishBeanFactoryInitialization(beanFactory);初始化剩下的单实例bean;
 *   			1)、先创建所有的单实例bean;getBean();
 *   			2)、获取所有创建好的单实例bean,判断是否是SmartInitializingSingleton类型的;
 *   				如果是就调用afterSingletonsInstantiated();
 * 		
 * 
 *
 */

-------------------------------------------------------------------

BeanFactoryPostProcesser

实例代码:

@Component
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {

	@Override
	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));
	}

}

----------------------------------------------------------------------

测试:

我们会发现blue已经有了但是还没有创建对象呢。

执行的时机:

BeanFactoryPostProcessor:beanFactory的后置处理器;
在BeanFactory标准初始化之后调用,来定制和修改BeanFactory的内容;
所有的bean定义已经保存加载到beanFactory,但是bean的实例还未创建
BeanFactoryPostProcessor原理:
 1)、ioc容器创建对象
 2)、invokeBeanFactoryPostProcessors(beanFactory);
如何找到所有的BeanFactoryPostProcessor并执行他们的方法;
 1)、直接在BeanFactory中找到所有类型是BeanFactoryPostProcessor的组件,并执行他们的方法
 2)、在初始化创建其他组件前面执行

debug:

拿到bean:

点进去:

执行方法:

---------------------------------------------39------------------------------------------------

下一个:BeanDefinitionRegitryPostProcesser它继承了BeanFactoryPostProcesser

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

	@Override
	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实例;
	@Override
	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);
	}

}

这句话重要:

测试:

原理:

这一步:

一直点进去会有这个方法,先从容器中获取所有的组件:

点进去:

依次触发。

在来到2的方法:

在最初的再往下翻还会有的:

-----------------------------------------------------------------40---------------------------------------------------------------------

ApplicationListener

代码:注意这个触发器是怎么写的?容器的事件。

@Component
public class MyApplicationListener implements ApplicationListener<ApplicationEvent> {
	//当容器中发布此事件以后,方法触发
	@Override
	public void onApplicationEvent(ApplicationEvent event) {
		// TODO Auto-generated method stub
		System.out.println("收到事件:"+event);
	}
}

测试:

事件的开发:

第一步:写监听器监听事件

第二步:监听器放在容器中

第三步:监听

---------------------------------------------------------------------------

我们自己如何发布事件?

我发布事件让监听器去监听。

-------------------41---------------------

我们看收到了三个事件:

*  【事件发布流程】:
 *  	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);
 *  		
 *   SmartInitializingSingleton 原理:->afterSingletonsInstantiated();
 *   		1)、ioc容器创建对象并refresh();
 *   		2)、finishBeanFactoryInitialization(beanFactory);初始化剩下的单实例bean;
 *   			1)、先创建所有的单实例bean;getBean();
 *   			2)、获取所有创建好的单实例bean,判断是否是SmartInitializingSingleton类型的;
 *   				如果是就调用afterSingletonsInstantiated();

源码:

1.第一步:

2.第二步:

容器刷新的event。

3.第三步:第一步get是获取多播器。

4.第四步

派发事件,拿到多播器,进而拿到所有的监听器。

这里可以异步也可以同步的。

5.进入invokeListener,listener拿到回调方法

---------------------    

事件的派发器是如何拿到的呢?

点进去:先找有没有没有就自动new一个。

获得多播器,没有就自己new一个多播器。

-------------------------------------------------------

如何知道容器有哪些监听器?这里使用的,就是这里。

在这里判断:

注册监听器:

在这里:

-------------------------------------------------------------------42--------------------------------------------------------------------

@EventListener

还可以这么监听:

@Service
public class UserService {
	
	@EventListener(classes={ApplicationEvent.class})// 指定监听什么事件
	public void listen(ApplicationEvent event){
		System.out.println("UserService。。监听到的事件:"+event);
	}
}

EventListenerMethodProcessor:

点进去:

单实例bean全部创建完成之后。

*   SmartInitializingSingleton 原理:->afterSingletonsInstantiated();
 *   		1)、ioc容器创建对象并refresh();
 *   		2)、finishBeanFactoryInitialization(beanFactory);初始化剩下的单实例bean;
 *   			1)、先创建所有的单实例bean;getBean();
 *   			2)、获取所有创建好的单实例bean,判断是否是SmartInitializingSingleton类型的;
 *   				如果是就调用afterSingletonsInstantiated();

第一步:

第二步:点进去

再点进去

第三步:我们再往前看下

创建对象。

第四步:

两次for循环第一次创建bean,第二次判断是不是SmartInitializingSingleton。

-------------------43---------------------

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip 【备注】 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用!有问题请及时沟通交流。 2、适用人群:计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、自动化、电子信息等)在校学生、专业老师或者企业员工下载使用。 3、用途:项目具有较高的学习借鉴价值,不仅适用于小白学习入门进阶。也可作为毕设项目、课程设计、大作业、初期项目立项演示等。 4、如果基础还行,或热爱钻研,亦可在此项目代码基础上进行修改添加,实现其他不同功能。 欢迎下载!欢迎交流学习!不清楚的可以私信问我! 毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值