spring 5.0.x源码学习系列十: 观察者设计模式与Spring 事件驱动模型


theme: channing-cyan

前言

  • 上篇博客spring 5.0.x源码学习系列九: FactoryBean和BeanFactory主要介绍了spring中的两个特殊的bean, FactoryBeanBeanFactory。本次,咱们将开始介绍下spring的事件驱动模型以及它使用的设计模式: 观察者设计模式

一、观察者模式

1.1 根据一个案例实现一个观察者设计模式

  • 假设有这么一个场景,小黄和小杨一起在看电影,当电影播放到精彩的剧情时,小黄小杨分别会有不同的行为表现。小黄会睡觉,小杨会哭泣。下面我们将使用代码来模拟出这个案例。

1.1.1 版本1

  • Application.java入口类


    在这里插入图片描述
  • Huang.java小黄实体类


    在这里插入图片描述
  • Yang.java小杨实体类


    在这里插入图片描述
  • Movie.java电影类


    在这里插入图片描述
  • 运行结果


    在这里插入图片描述
  • 小结:
    • 使用上述的方式实现了案例的需求,但有如下缺点
    1. 观察者需要全程观看电影,占用观察者大量时间
    2. 当电影播放到精彩部分的时候自己来主动通知自己要做些事情了。

1.1.2 版本二

  • 修复了版本一的问题:

    1. 由电影来通知观察者,省去观察者需要全程观看电影的缺点,由电影来通知观察者该做一些事情了
  • Application.java


    在这里插入图片描述
  • Huang.java


    在这里插入图片描述
  • Yang.java


    在这里插入图片描述
  • Movie.java


    在这里插入图片描述
  • 运行结果


    在这里插入图片描述
  • 小结,版本二的观察者模式具有如下缺点:

    1. 观察者的行为动作不统一,被观察者代码维护难度大
    2. 扩展性低,若观察者需要增加,则需要手动在被观察中添加观察者的引用,重复工作多

1.1.3 版本3

  • 此版本修复了版本2的缺点:
    1. 观察者行为动作不统一
    2. 扩展性低
  • MovieObserver.java


    在这里插入图片描述
  • Application.java


    在这里插入图片描述
  • Huang.java


    在这里插入图片描述
  • Movie.java


    在这里插入图片描述
  • Yang.java
    在这里插入图片描述
  • 运行结果


    在这里插入图片描述
  • 小结
    1. 版本3是观察者模式的基本模型,但是它并没有抽象出事件的概念。其实也抽象出来了。MovieObserver相当于事件的监听者,Movie播放精彩部分的动作是一个事件。notifyObservers相当于发布事件。

1.1.4 版本4

  • 此版本基于版本3做了些修改, 将事件、监听者、发布事件的概念添加了进去

  • MovieEvent.java


    在这里插入图片描述
  • MovieListener.java


    在这里插入图片描述
  • Application.java


    在这里插入图片描述
  • Huang.java


    在这里插入图片描述
  • Movie.java


    在这里插入图片描述
  • Yang.java


    在这里插入图片描述
  • 运行结果


    在这里插入图片描述
  • 小结

    1. 版本4将事件的概念抽象了出来,会根据发布不同的事件响应不同的行为
    2. 在此版本中,一共定义了两种类型的事件感人类型事件, 喜剧类型事件

1.2 观察者设计模式特点

  • 被观察者持有观察者的引用
  • 耦合性低,被观察者在特定时间发布特定事件,观察者具体的行为动作由观察者自己决定
  • 提供新增和移除观察者api,可任意扩容观察者数量

二、Spring 事件模型

2.1 spring事件模型类型

  • ApplicationContextEvent
  • RequestHandlerEvent

2.2 ApplicationContextEvent事件触发步骤

  1. 新建自定义事件源。继承ApplicationEvent类, 并重写带参构造方法

       public class MyApplicationEvent extends ApplicationEvent {
           // 带参构造方法中的参数就是可以为事件添加一些参数, 即
           // 自定义一些事件
           public MyApplicationEvent(Object source) {
               super(source);
           }
       }
    
  2. 新建事件监听器。

       // 1. 需要添加@Component注解, 让监听者作为spring的一个bean
       // 2. 其次, 要实现ApplicationListener接口, 泛型为上述定义的事件源(MyApplicationEvent)
       @Component
       public class MyApplicationListener implements ApplicationListener {
       
           @Override
           public void onApplicationEvent(MyApplicationEvent event) {
               System.out.println("监听者, " + event);
           }
       }
    
  3. 获取spring ApplicationEventPublisher对象

       // 新建了一个bean来维护上下文对象
       @Component
       public class ApplicationContextHolder implements ApplicationContextAware {
       
           private static ApplicationContext applicationContext;
       
           @Override
           public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
               applicationContext = applicationContext;
           }
      
           public static ApplicationContext getApplicationContext() {
               return applicationContext;
           }
       }
    
  4. 触发事件

       // 拿到spring上下文发布事件
       // spring上下文如何拿到? spring上下文是指实现了ApplicationContext接口的类
       // 在以java config技术开启的spring应用程序中, 可以用AnnotationConfigApplicationContext的实例, 第三步可以忽略
       // 若是spring web应用程序, eg: spring mvc, 则可以新增一个bean并实现ApplicationContextAware接口, eg: 第三步
       ApplicationContextHolder.getApplicationContext().publishEvent(new MyApplicationEvent(new String[] {
          "1", "2"
       }))
    

2.3 ApplicationContextEvent的四种类型总结

类型 如何触发 触发时机 spring 上下文初始化后是否能触发
ContextRefreshEvent 发布类型为ContextRefreshEvent的事件. 调用spring上下文refersh方法
ContextStopEvent 发布类型为ContextStopEvent的事件 调用spring上下文stop方法
ContextStartEvent 发布类型为ContextStartEvent的事件 调用spring上下文start方法
ContextCloseEvent 发布类型为ContextCloseEvent的事件 调用spring上下文close方法
  • 上述的触发时机可参考此url:https://github.com/AvengerEug/observer-csdn/blob/develop/src/main/java/springevent/Entry.java

2.4 RequestHandlerEvent类型的事件

  • 待总结

三、总结

  • Spring事件驱动模型是基于观察者设计模式衍生而来的,jdk也有自带的观察者设计模式类,分别是被观察者: Observable观察者: Observer。但jdk自带的观察者设计模式类并没有把事件集成进去,虽然JDK中存在事件相关的类,比如EventObject类,但仅仅是被应用到了java swing编程。
  • 本次博客代码仓库地址: https://github.com/AvengerEug/observer-csdn.git
  • I am a slow walker, but I never walk backwards.

你可能感兴趣的:(spring 5.0.x源码学习系列十: 观察者设计模式与Spring 事件驱动模型)