Spring中的事件驱动模型(二)

时间:2022-05-06
本文章向大家介绍Spring中的事件驱动模型(二),主要内容包括前文回顾、应用Spring中的事件驱动模式、定义监听器、测试类、总结、基本概念、基础应用、原理机制和需要注意的事项等,并结合实例形式分析了其使用技巧,希望通过本文能帮助到大家理解应用这部分内容。

前文回顾

前一篇文章讲了Spring中的事件驱动模型相关概念。重点篇幅介绍了Spring的事件机制,Spring的事件驱动模型由事件、发布者和订阅者三部分组成,结合Spring的源码分析了这三部分的定义与实现。本文主要结合具体例子讲解Spring中的事件驱动。笔者在写Spring Cloud Bus中的事件的订阅与发布两篇文章的时候,想到要把Spring中的事件驱动模型的讲解给补充一下,这块也是属于更加基础的知识点。

应用Spring中的事件驱动模式

我们示例配置信息的刷新,当配置服务器收到提交的配置事件之后,将会触发各个服务响应的更新自己的配置。具体代码如下:

事件

1public class ConfigRefreshEvent extends ApplicationEvent {
2    public ConfigRefreshEvent(final String content) {
3        super(content);
4    }
5}

定义一个配置刷新的事件。继承ApplicationEvent即可,content即为需要传递的object。

定义监听器

定义两个服务,都实现了SmartApplicationListener,该接口继承自ApplicationListenerOrdered接口,属于标准监听器的扩展接口,公开更多的元数据,例如支持的事件类型和可排序的监听器。

 1public interface SmartApplicationListener extends ApplicationListener<ApplicationEvent>, Ordered {
 2
 3    /**
 4     * 决定该监听器是否支持给定的事件
 5     */
 6    boolean supportsEventType(Class<? extends ApplicationEvent> eventType);
 7
 8    /**
 9     * 决定该监听器是否支持给定的目标类型,支持才会调用
10     */
11    boolean supportsSourceType(Class<?> sourceType);
12
13}

下面贴出两个Service的代码。

ServiceAListener

 1@Component
 2public class ServiceAListener implements SmartApplicationListener {
 3
 4    @Override
 5    public boolean supportsEventType(final Class<? extends ApplicationEvent> eventType) {
 6        return eventType == ConfigRefreshEvent.class;
 7    }
 8
 9    @Override
10    public boolean supportsSourceType(final Class<?> sourceType) {
11        return sourceType == String.class;
12    }
13
14    @Override
15    public void onApplicationEvent(final ApplicationEvent event) {
16        System.out.println("ServiceA收到新的配置:" + event.getSource());
17    }
18
19    @Override
20    public int getOrder() {
21        return Ordered.HIGHEST_PRECEDENCE;
22    }
23} 

ServiceBListener

 1@Component
 2public class ServiceBListener implements SmartApplicationListener {
 3
 4    @Override
 5    public boolean supportsEventType(final Class<? extends ApplicationEvent> eventType) {
 6        return eventType == ConfigRefreshEvent.class;
 7    }
 8
 9    @Override
10    public boolean supportsSourceType(final Class<?> sourceType) {
11        return sourceType == String.class;
12    }
13
14    @Override
15    public void onApplicationEvent(final ApplicationEvent event) {
16        System.out.println("ServiceB收到新的配置:" + event.getSource());
17    }
18
19    @Override
20    public int getOrder() {
21        return Ordered.LOWEST_PRECEDENCE;
22    }
23}

测试类

 1@RunWith(SpringRunner.class)
 2@SpringBootTest
 3public class ApplicationTest {
 4    @Autowired
 5    private ApplicationContext applicationContext;
 6
 7    @Test
 8    public void testPublishEvent() {
 9        System.out.println("发布配置更新:");
10        ConfigRefreshEvent event = new ConfigRefreshEvent("配置信息更新了")
11        applicationContext.publishEvent(event);
12    }
13}

上面是我们的测试类,相当于事件的发布者,首先定义一个配置刷新的事件,然后通过注入的ApplicationContext发布该事件。

由于ServiceA的优先级高于ServiceB,所以我们看到如下的结果:

1发布配置更新:
2ServiceA收到新的配置:配置信息更新了
3ServiceB收到新的配置:配置信息更新了

总结

本文比较简单,在上一篇介绍Spring中的事件驱动模型基础上,具体应用到配置刷新的场景中。Spring的事件驱动模型使用的是观察者模式。通过ApplicationEvent抽象类和ApplicationListener接口,可以实现事件的定义与监听,ApplicationContext则实现了事件的发布。例子中使用的SmartApplicationListener扩展了标准的事件监听接口,监听器在处理Event时,可以对传入的Event进行判断,并且可以设定监听器的优先级。后面抽时间会写一下 Spring Cloud 的热更新机制,也是基于Spring中的事件驱动模型。