前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Spring全注解开发--扩展原理

Spring全注解开发--扩展原理

作者头像
大忽悠爱学习
发布2021-11-15 15:05:02
3510
发布2021-11-15 15:05:02
举报
文章被收录于专栏:c++与qt学习c++与qt学习

扩展原理


BeanFactoryPostProcessor:beanFactory的后置处理器

BeanPostProcessor:bean的后置处理器,bean创建对象初始化前后进行拦截工作的

BeanFactoryPostProcessor:beanFactory的后置处理器,在BeanFactory标准初始化之后调用,来定制和修改BeanFactory里面的内容

所有的bean定义已经保存加载到beanFactory,但是bean的实例还未创建

在这里插入图片描述
在这里插入图片描述

继承BeanFactoryPostProcessor 来进行测试:

代码语言:javascript
复制
@Component
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory configurableListableBeanFactory) throws BeansException {
        int count = configurableListableBeanFactory.getBeanDefinitionCount();
        System.out.println("postProcessBeanFactory执行中...");
        System.out.println("bean的数量:"+count);
        String[] beans = configurableListableBeanFactory.getBeanDefinitionNames();
        System.out.println(Arrays.asList(beans));
    }
}
在这里插入图片描述
在这里插入图片描述

BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor

代码语言:javascript
复制
public interface BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor {
    void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry var1) throws BeansException;
}

postProcessBeanDefinitionRegistry

调用时机:在所有bean定义信息将要被加载,bean实例还未创建时

BeanDefinitionRegistry: bean定义信息的保存中心

以后BeanFactory就是按照BeanDefinitionRegistry里面保存的每一个bean定义信息创建bean实例

实现BeanDefinitionRegistryPostProcessor 接口进行测试:

代码语言:javascript
复制
@Component
public class MyBeanDefinitonRegistry implements BeanDefinitionRegistryPostProcessor {
    //BeanDefinitionRegistry: bean定义信息的保存中心
    //以后BeanFactory就是按照BeanDefinitionRegistry里面保存的每一个bean定义信息创建bean实例*
    @Override
    public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry beanDefinitionRegistry) throws BeansException {
        //给容器中注册bean
        //法1:
       // RootBeanDefinition rootBeanDefinition = new RootBeanDefinition(department.class);
        //法2:
        AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(department.class).getBeanDefinition();
        beanDefinitionRegistry.registerBeanDefinition("depart",beanDefinition);
        System.out.println("1.bean的数量:"+beanDefinitionRegistry.getBeanDefinitionCount());
    }

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory configurableListableBeanFactory) throws BeansException {
        System.out.println("2.bean的数量:"+configurableListableBeanFactory.getBeanDefinitionCount());
    }
}
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

ApplicationListener:监听容器中发布的事件,事件驱动模型开发

代码语言:javascript
复制
@FunctionalInterface
public interface ApplicationListener<E extends ApplicationEvent> extends EventListener {
    void onApplicationEvent(E var1);
}

ApplicationListener负责监听ApplicationEvent及其下面的子事件

继承ApplicationListener接口,来监听事件,记得将组件放入容器中:

代码语言:javascript
复制
@Component
public class MyApplicationListener implements ApplicationListener {
    //当容器中发布事件后,方法触发
    @Override
    public void onApplicationEvent(ApplicationEvent applicationEvent) {
        System.out.println("收到事件"+applicationEvent);
    }
}

默认只会有两个事件,即容器刷新事件和容器关闭事件

在这里插入图片描述
在这里插入图片描述

或者使用@EventListener注解来定义一个监听器

注:@EventListener(classes={ApplicationEvent.class})中classes监听的类可以是多个,以逗号分开

这里如果使用注解,还是注意当前所在类要加入到容器中

代码语言:javascript
复制
@ComponentScan({"com.config"})
@Configuration
public class MyConfig
{
    @EventListener(classes={ApplicationEvent.class})
    public void listen(ApplicationEvent event){
        System.out.println("收到事件"+event);
    }
}
在这里插入图片描述
在这里插入图片描述

自己发布事件:

代码语言:javascript
复制
public class Main
{
  //传入的是配置类的位置,一开始是加载配置类,之前是加载配置文件的位置
  private static AnnotationConfigApplicationContext ioc= new AnnotationConfigApplicationContext(MyConfig.class);

  public static void main(String[] args)
  {
    //发布事件
    ioc.publishEvent(new ApplicationEvent(new String("我发布的一个事件")) {
    });
    ioc.close();
  }
}
在这里插入图片描述
在这里插入图片描述


原理:

代码语言:javascript
复制
事件: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();
 *         

二、实例
ApplicationListener:监听容器中发布的事件。事件驱动模型开发;
 *       public interface ApplicationListener<E extends ApplicationEvent>
 *         监听 ApplicationEvent 及其下面的子事件;
 * 
 *      步骤:
 *         1)、写一个监听器(ApplicationListener实现类)来监听某个事件(ApplicationEvent及其子类)
 *             @EventListener;
 *             原理:使用EventListenerMethodProcessor处理器来解析方法上的@EventListener;
 * 
 *         2)、把监听器加入到容器;
 *         3)、只要容器中有相关事件的发布,我们就能监听到这个事件;
 *                 ContextRefreshedEvent:容器刷新完成(所有bean都完全创建)会发布这个事件;
 *                 ContextClosedEvent:关闭容器会发布这个事件;
 *         4)、发布一个事件:
 *                 applicationContext.publishEvent();
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2021-08-09 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 扩展原理
  • BeanFactoryPostProcessor:beanFactory的后置处理器
  • BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor
  • ApplicationListener:监听容器中发布的事件,事件驱动模型开发
    • 自己发布事件:
      • 原理:
      相关产品与服务
      容器服务
      腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档