• Spring观察者模式 - 事件监听机制(发布订阅模式)


    目录

    应用场景举例:

    观察者模式的优点

    观察者模式的缺点

    Spring中的观察者模式

    代码实现

    1、定义事件类型

    2、监听者 - 短信

    3、监听者 - 邮件

    4、发布者 - 发布事件

    5、测试发布-监听


    应用场景举例:

          · 一个事件多个处理方式的场景。

       · 当用户下单成功后会发送手机短信、邮箱、微信、等。

    观察者模式的优点

    • 观察者和被观察者之间是抽象耦合,不管是增加观察者还是被观察者都非常容易扩展。
    • 建立一套触发机制。

    观察者模式的缺点

    • 观察者模式需要考虑开发效率和运行效率问题,一个被观察者,多个观察者,开发和调试比较复杂,Java消息的通知默认是顺序执行的,一个观察者卡壳,会影响整体的执行效率。这种情况一般考虑异步的方式

    Spring中的观察者模式

    ApplicationContext中事件处理是由ApplicationEvent类和ApplicationListener接口来提供的。如果一个Bean实现了ApplicationListener接口,并且已经发布到容器中去,每次ApplicationContext发布一个ApplicationEvent事件,这个Bean就会接到通知。Spring事件机制是观察者模式的实现。

    Spring中的事件机制涉及到者几个类文件 :

       ApplicationListener 监听者事件监听)接口

       ApplicationEventPublisherAware 发布者(事件发布类)接口

       ApplicationEvent 事件类型(定义事件的数据结构)类

    代码实现

    1、定义事件类型

    就是定义事件的数据结构。

    1. /**
    2. * 定义事件类型(定义事件的数据结构)
    3. */
    4. public class UserRegisterEvent extends ApplicationEvent {
    5. /** 用户名 */
    6. private String username;
    7. public UserRegisterEvent(Object source) {
    8. super(source);
    9. }
    10. public UserRegisterEvent(Object source, String username) {
    11. super(source);
    12. this.username = username;
    13. }
    14. public String getUsername() {
    15. return username;
    16. }
    17. }

    2、监听者 - 短信

    1. /**
    2. * 监听者 - 短信监听(异步执行)
    3. */
    4. @Component
    5. public class SmsListener implements ApplicationListener {
    6. @Override
    7. @Async //异步
    8. public void onApplicationEvent(UserRegisterEvent event) {
    9. System.out.println(Thread.currentThread() + ",短信监听到>>>" + event.getUsername()+ "," + event.getSource());
    10. }
    11. }

    3、监听者 - 邮件

    1. @Component
    2. public class EmailListener implements ApplicationListener {
    3. private static final Logger logger = Logger.getLogger(EmailListener.class);
    4. @Override
    5. @Async
    6. public void onApplicationEvent(OrderEvent event) {
    7. System.out.println(Thread.currentThread() + ",邮件监听到>>>" + event.getMessage()+ "," + event.getSource());
    8. }
    9. }

    4、发布者 - 发布事件

    1. /**
    2. * 发布者
    3. */
    4. @Service
    5. public class UserServicePublisher implements ApplicationEventPublisherAware {
    6. private ApplicationEventPublisher applicationEventPublisher;
    7. @Override
    8. public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {
    9. this.applicationEventPublisher = applicationEventPublisher;
    10. }
    11. /**
    12. * 发布
    13. */
    14. public void register(String username) {
    15. System.out.println("发布 >>>>>> "+ username);
    16. applicationEventPublisher.publishEvent(new UserRegisterEvent(this, username));
    17. }
    18. }

    最后在启动类上加上开启异步注解 @EnableAsync,否则监听者将会串行执行。

    5、测试发布-监听

    1. @RestController
    2. public class TestController {
    3. @Autowired
    4. private UserServicePublisher userService;
    5. @RequestMapping("/test/register")
    6. public void testRegister(){
    7. userService.register("王明");
    8. }
    9. }
    请求:http://127.0.0.1:8080/test/register 
    然后查看控制台打印:
    

     

  • 相关阅读:
    一位程序员感慨:互联网行业太过共享,才导致了门槛越来越低
    LSTMviz配置
    【html5期末大作业】基于HTML+CSS+JavaScript管理系统页面模板
    面试算法十问2(中英文)
    深度学习入门(二十六)卷积神经网络——池化层
    网址收藏-技术类
    经典供货保密协议模板
    电影大师杂记
    一篇文章掌握整个JVM,JVM超详细解析!!!
    matlab 求数列极限
  • 原文地址:https://blog.csdn.net/qq_36881887/article/details/127666814