当前位置:   article > 正文

实战揭秘:深入解析SSE结合EventBus实现消息定向推送_sse通知推送

sse通知推送

        

目录

一、EventBus 简介

        1.1 EventBus 工作流程

二、SSE 基于 EventBus 实现定向推送

        2.1 使用 SSE 注意事项


        SSE(Server-Sent Events)是指服务器发送事件,这是一种浏览器 API,允许服务器端向客户端持续推送数据,而无需客户端发起请求。

        比如在社交媒体更新、新闻直播、股票市场、物联网智能设备等等方面,如下图。如果数据发生了更新,更新的数据能实时高效的推送给客户端就显的尤为重要。传统的方式是客户端需要去主动获取数据的变化,但是这种事不实时的,效率低,而且浪费资源。

        

        相比于 WebSocket,SSE 比较简单,基于 HTTP 协议,易于理解,但是 SSE 也有一个天然的不足,那就是无法定向推送数据,推送数据时广播的模式,所有的客户端都能收到。

一、EventBus 简介

        EventBus 是一个广泛应用于开发中的轻量级事件发布/订阅框架,它的核心设计理念是简化应用程序内部各组件间的通信。通过采用发布/订阅(Publish/Subscribe)设计模式,EventBus能够有效地降低组件之间的耦合度,提高代码可读性和维护性。

        1.1 EventBus 工作流程

        EventBus 的工作流程如下:

  1. 定义事件(Event):首先,需要自定义事件类,这个类可以封装任何类型的数据,比如行为数据、数据更新或者其他需要传递的信息。
  2. 注册订阅者(Subscriber Registration):使用注解 @Subscribe 标记订阅者类中的某个方法,表示该方法想要接收某种类型的事件。在初始化阶段,需要讲订阅者注册到 EventBus 中,这样 EventBus 就能知道哪些对象对何种事件感兴趣。
  3. 发布事件(Post):当有事件需要传递时,发布者通过调用 post 方法,将事件发布到 EventBus 上。
  4. 事件分发(Dispatching):EventBus 收到事件后,会根据事件类型找到所有注册了该事件类型处理方法的订阅者。根据事件订阅时指定的线程模式,EventBus 会选择合适的线程执行订阅者的方法。线程模型如下:
    1. POSTING:事件处理在发布事件所在线程中执行。
    2. MAIN:如果不在主线程,则切换到主线程执行事件处理。
    3. MAIN_ORDERED:类似MAIN,但在主线程中按照事件发布的顺序逐个执行。
    4. BACKGROUND:如果不在后台线程,则新建一个后台线程执行。
    5. ASYNC:无论在哪种线程环境下,都会在独立的线程池中异步执行。
  5. 执行订阅者方法:EventBus 通过反射调用订阅者中对应注解的方法,并将发布的事件对象作为参数传递给该方法。

二、SSE 基于 EventBus 实现定向推送

        由于 SSE 只能实现广播模式的消息推送,如果要实现推送数据到指定的客户端,就需要做一些改动,加入有这么一个场景,某一个客户端关注某一个事件,当该事件发生变动时,只把这个新时间推送给指定的客户端,这如何实现呢?

        在客户端的入参中需要加入 clientId 来进行区分,新数据到来后也要有相应的标识能获取到绑定的客户端,这样就能将变动的数据推送给指定的客户端了,具体实现如下。

        添加依赖

  1. // Spring boot相关依赖请自行添加
  2. <dependency>
  3. <groupId>com.google.code.findbugs</groupId>
  4. <artifactId>jsr305</artifactId>
  5. <version>3.0.2</version>
  6. </dependency>
  7. <dependency>
  8. <groupId>com.google.guava</groupId>
  9. <artifactId>guava</artifactId>
  10. <version>32.0.1-jre</version>
  11. </dependency>

        添加 EventBus 配置类

  1. package cn.scf.sse.config;
  2. import com.google.common.eventbus.EventBus;
  3. import org.springframework.context.annotation.Bean;
  4. import org.springframework.context.annotation.Configuration;
  5. @Configuration
  6. public class EventBusConfig {
  7. @Bean
  8. public EventBus eventBus() {
  9. return new EventBus();
  10. }
  11. }

        定义事件类

  1. package cn.scf.sse.event;
  2. public class ClientEvent {
  3. // 客户端ID
  4. private final String clientId;
  5. private final String data;
  6. public ClientEvent(String clientId, String data) {
  7. this.clientId = clientId;
  8. this.data = data;
  9. }
  10. public String getClientId() {
  11. return clientId;
  12. }
  13. public String getData() {
  14. return data;
  15. }
  16. }

        定义 SSE 的管理类

  1. package cn.scf.sse.event;
  2. import com.google.common.eventbus.EventBus;
  3. import lombok.extern.slf4j.Slf4j;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.stereotype.Component;
  6. import org.springframework.web.servlet.mvc.method.annotation.SseEmitter;
  7. import java.util.Map;
  8. import java.util.concurrent.ConcurrentHashMap;
  9. @Slf4j
  10. @Component
  11. public class SSEManager {
  12. // 存储与客户端关联的SseEmitter实例
  13. private final Map<String, SseEmitter> clientEmitters = new ConcurrentHashMap<>();
  14. @Autowired
  15. private EventBus eventBus;
  16. public void register(Object listener) {
  17. // 注册监听器,当EventBus上有事件时,根据事件中的客户端标识发送数据
  18. eventBus.register(listener);
  19. }
  20. public void postEvent(String clientId, String data) {
  21. ClientEvent event = new ClientEvent(clientId, data);
  22. eventBus.post(event);
  23. }
  24. // 关闭连接时调用
  25. public void closeEmitter(String clientId, String exception) {
  26. SseEmitter sseEmitter = clientEmitters.remove(clientId);
  27. if (sseEmitter != null) {
  28. sseEmitter.complete();
  29. }
  30. log.info("关闭连接清理资源成功, ex: {}", exception);
  31. }
  32. public Map<String, SseEmitter> getClientEmitters() {
  33. return clientEmitters;
  34. }
  35. }

        事件处理类,从事件总线订阅了自己感兴趣的事件

  1. package cn.scf.sse.event;
  2. import com.alibaba.fastjson.JSONObject;
  3. import com.google.common.eventbus.Subscribe;
  4. import lombok.extern.slf4j.Slf4j;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.stereotype.Service;
  7. import org.springframework.web.servlet.mvc.method.annotation.SseEmitter;
  8. import javax.annotation.PostConstruct;
  9. import java.util.Map;
  10. @Service
  11. @Slf4j
  12. public class EventHandler {
  13. @Autowired
  14. private SSEManager sseManager;
  15. @PostConstruct
  16. public void init() {
  17. // 初始化时将自己注册到 EventBus 中
  18. sseManager.register(this);
  19. }
  20. @Subscribe
  21. public void handleEvent(ClientEvent event) {
  22. Map<String, SseEmitter> clientEmitters = sseManager.getClientEmitters();
  23. SseEmitter emitter = clientEmitters.get(event.getClientId());
  24. if (emitter != null) {
  25. new Thread(() -> {
  26. try {
  27. String message = JSONObject.toJSONString(event);
  28. // 发送给客户端
  29. emitter.send(SseEmitter.event().data(message));
  30. } catch (Exception e) {
  31. emitter.completeWithError(e);
  32. }
  33. }).start();
  34. }
  35. }
  36. }

        SSE 接口实现

  1. @RestController
  2. public class SseController {
  3. @Autowired
  4. private SSEManager sseManager;
  5. @CrossOrigin
  6. @GetMapping(value = "/sse/{clientId}", produces = MediaType.TEXT_EVENT_STREAM_VALUE)
  7. public SseEmitter handleSseRequest(@PathVariable String clientId) {
  8. // 设置超时时间, 如果在指定的时间内没有向客户端发送任何数据,则连接将自动关闭。
  9. SseEmitter emitter = new SseEmitter(TimeUnit.MINUTES.toMillis(10));
  10. Map<String, SseEmitter> clientEmitters = sseManager.getClientEmitters();
  11. clientEmitters.put(clientId, emitter);
  12. // 当SSE连接关闭时,从管理器中移除
  13. emitter.onCompletion(() -> sseManager.closeEmitter(clientId, null));
  14. // 监听链接错误
  15. emitter.onError((ex) -> sseManager.closeEmitter(clientId, ex.getMessage()));
  16. return emitter;
  17. }
  18. // 此接口用来模拟有新数据到来时发布事件,将新数据推送给指定客户端,实际中这里可能是三方数据推送,也可能是数据库数据变化
  19. @GetMapping(value = "/pub/event")
  20. public String pubEvent(@RequestParam String clientId) {
  21. sseManager.postEvent(clientId, "发送事件客户端ID:" + clientId + "-" +UUID.randomUUID().toString());
  22. return "ok";
  23. }
  24. }

        最后是 H5 调用 SSE 接口

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <title>SSE Example</title>
  5. <script type="text/javascript">
  6. document.addEventListener('DOMContentLoaded', function() {
  7. let clientId = 'some-client-id'; // 假设已经获取到客户端标识
  8. let source = new EventSource(`http://localhost:8080/sse/${clientId}`);
  9. source.onmessage = function(event) {
  10. let data = event.data;
  11. console.log(data);
  12. // 更新页面上的某些内容,例如:
  13. document.getElementById('sse').innerHTML += data + '<br>';
  14. };
  15. source.onerror = function(event) {
  16. console.error("EventSource failed.");
  17. };
  18. // source.close();
  19. });
  20. </script>
  21. </head>
  22. <body>
  23. <h1>Server Sent Events</h1>
  24. <div id="sse"></div>
  25. <!-- 页面内容根据接收到的数据更新 -->
  26. </body>
  27. </html>

        当服务启动时,EventHandler 就将自己注册到了 EventBus 中,并通过 @Subscribe 标记想要监听的事件,在方法内部取出客户端与 SseEmitter 的关系,判断是否有对应的 SseEmitter,如果存在,就向客户端推送新的数据。

        2.1 使用 SSE 注意事项

        使用Server-Sent Events(SSE)时,以下是一些值得注意的关键事项:

        浏览器兼容:SSE 是 HTML5 的一项功能,所以并非所有浏览器都支持。在使用之前,应检查目标浏览器是否支持 EventSource API。

        连接管理:SSE 通过单个 HTTP 连接进行数据推送,这意味着浏览器会维持一个长连接至服务器。确保服务器端正确处理连接的生命周期,包括维持连接、处理空闲连接、以及在连接断开时自动重新连接。

        并发限制:浏览器可能对同一域名下的并发 SSE 连接有所限制,通常每个浏览器标签页共享一个最大连接数,超过这个数量的 SSE 连接可能无法建立。

        资源管理:由于连接长期存在,需要考虑服务器资源消耗和客户端内存占用。在服务器端确保及时释放不再使用的资源,客户端也需要适当管理 EventSource 对象,比如在页面卸载时取消注册事件源。

        

往期经典推荐

实时数据传输的新里程——Server-Sent Events(SSE)消息推送技术-CSDN博客

Redis字符创类型内存消耗的奥秘-CSDN博客

系统优化都没做过?看这篇就够了-CSDN博客

深入浅出 Drools 规则引擎-CSDN博客

SpringBoot开箱即用魔法:深度解析与实践自定义Starter-CSDN博客

声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号