首页 > 其他分享 >实时数据传输的新里程——Server-Sent Events(SSE)消息推送技术

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

时间:2024-03-16 11:59:06浏览次数:17  
标签:HTTP 实时 Server SSE 股票价格 推送 Events 客户端

目录

一、背景介绍

二、SSE场景案例

三、工作原理

        3.1 SSE 工作流程

        3.2 工作原理


一、背景介绍

        传统的请求模型是客户端发送一个请求到服务端,然后服务端做相应的处理,然后再将结果返回给客户端,这种“一问一答”的方式随着时间的推移并不能适应所有的场景。

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

        为了解决这一问题,Server-Sent Event(SSE)作为一种基于 HTTP 长连接的单向通信技术应用而生。相较于 WebSocket的不同之处如下

  • 单向双向:SSE只支持服务端到客户端的单向数据流,而 WebSocket 支持全双工通信;
  • 易用性:SSE 在实现上比 WebSocket 简单,尤其对于只需要单向通信的场景;
  • 兼容性:SSE 可以再任何支持 HTTP 的平台上使用,而 WebSocket 需要特定的服务器和客户端支持,HTTP 协议简单易于理解。

二、SSE场景案例

        假设我们开发一个股票价格实时监控系统,用户关注的股票价格发生变动时,要实时的推送给客户端。下面用 SpringBoot 来实现该场景。

        定义一个接口,用来实现获取股票价格和实时推送数据。

@RestController
public class StockController {

 
    @CrossOrigin
    @GetMapping(value = "/stock-price", produces = MediaType.TEXT_EVENT_STREAM_VALUE, name = "股票价格变动实时推送")
    public SseEmitter streamStockPrice() {
        SseEmitter emitter = new SseEmitter();
        Random random = new Random();
        new Thread(() -> {
            try {
                // 最新的股票价格,这里通过随机数模拟价格,实际上的股票价格要通过金融平台获取,比如消息中间件、开放API等
                double price = 100 + random.nextDouble() * 10;
                // 构造股票价格信息
                String message = String.format("%.2f", price);
                // 发送给客户端
                emitter.send(SseEmitter.event().data(message));
                TimeUnit.SECONDS.sleep(1);
            } catch (Exception e) {
                emitter.completeWithError(e);
            }
        }).start();
        System.out.println("发送消息");
        return emitter;
    }

        注意代码中的 MediaType.TEXT_EVENT_STREAM_VALUE 属性,它是 Spring 框架中预定义的一个常量,代表 MIME 类型“text/event-stream”,这时一种特殊的类型,用于 SSE 技术。设置该属性后意味着该方法将返回一个符合 SSE 规范的数据流,客户端可以正确地解析并处理这些数据流,进而实现实时数据更新。

        在方法内部,我们创建了一个 SseEmitter 对象作为事件发射器,并在一个单独的线程中不断生成随机的股票价格,并将价格转换为字符串形式发送给客户端。通过 emitter.send()方法发送的数据会被封装为SSE事件流的形式,客户端可以通过监听该事件流来实时接收股票价格。

        接下来,我们可以创建一个 HTML 页面来展示实时股票的价格。

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>实时股票价格监控</title>
</head>
<body>
<h1>实时股票价格</h1>
<div id="stock-price"></div>

<script>
    const eventSource = new EventSource('http://localhost:8080/stock-price');
    eventSource.onmessage = function (event) {
        document.getElementById('stock-price').innerHTML = event.data;
    };
</script>
</body>
</html>

        在 HTML 页面中,我们通过 new EventSource('/stock-price') 创建了一个 EventSource 对象,它会与 /stock-price 路径建立SSE连接。然后,我们通过 eventSource.onmessage 定义了接收消息的回调函数,在收到新消息时更新页面上的股票价格。

三、工作原理

        3.1 SSE 工作流程

        根据上图看下详细的工作流程

  1. 客户端初始化链接:客户端通过 JavaScript 创建一个新的 EventSource 对象,指定服务器的 URL 作为源地址。
  2. 服务器响应:当客户端请求该 URL 时,服务器会识别出这是一个 SSE 请求,并维持 HTTP连接的开启状态,返回一个 Content-Type为text/event-stream 的响应。响应体将以data:标签包裹的数据字段的形式不断发送更新事件。
  3. 数据推送:服务器在有新事件需要推送时,通过同一个 HTTP 连接向客户端发送事件数据。
  4. 客户端接收事件:客户端的EventSource对象监听各种事件,如messageopenerror。当服务器发送数据时,客户端会触发相应的事件处理函数,例如,每当接收到新的数据片段时,就会触发onmessage事件。
  5. 持久连接与错误处理:SSE 连接是持久的,除非网络中断或服务器关闭连接。如果连接意外断开,EventSource会自动尝试重新连接(根据retry头或默认间隔)

        3.2 工作原理

        SSE 主要利用了 HTTP 长连接特性,始终保持 TCP 连接,避免了传统轮询带来的延迟和资源浪费。服务器基于事件驱动,按需推送事件,客户端通过事件监听机制响应变化。

        总结来说,SSE 的优势还包括其简洁易用的 API,只需要在客户端使用 EventSource 接口即可处理服务器推送的事件,而服务器端则只需要按照特定的 text/event-stream 格式发送数据。同时,由于基于 HTTP 协议,SSE 具有很好的跨域支持和防火墙穿透能力,且在大多数现代浏览器中都有良好的支持。

往期经典推荐

从单核到多核:揭秘多核CPU的高效利用策略-CSDN博客

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

一文看懂Nacos如何实现高效、动态的配置中心管理_bootstrap.properties 配置nacos-CSDN博客

Tomcat架构究竟是什么?灵魂原来在这里-CSDN博客

Redis性能大挑战:深入剖析缓存抖动现象及有效应对的战术指南_redis 缓存抖动怎么解决-CSDN博客

标签:HTTP,实时,Server,SSE,股票价格,推送,Events,客户端
From: https://blog.csdn.net/qq_39209927/article/details/136733231

相关文章

  • 通过 Metrics Server 查看 Kubernetes 资源指标
    1.简介MetricsServer是一个用于Kubernetes集群的监控工具,它用于收集、存储和提供关于集群中各种资源的度量数据。MetricsServer是Kubernetes中一个核心的指标收集器,可以提供关于CPU和内存使用情况、节点资源利用率以及其他重要指标的信息。它主要用于水平自动扩展(Horiz......
  • Windows Server 2022 OVF, updated Feb 2024 (sysin) - VMware 虚拟机模板
    WindowsServer2022OVF,updatedFeb2024(sysin)-VMware虚拟机模板2024年2月版本更新,现在自动运行sysprep,支持ESXiHostClient部署请访问原文链接:https://sysin.org/blog/windows-server-2022-ovf/,查看最新版。原创作品,转载请保留出处。作者主页:sysin.org......
  • 深入了解WebAssembly:编译与测试指南
    在当今的Web开发中,WebAssembly(简称Wasm)作为一种新型的低级字节码格式,正在逐渐流行。它提供了一种高效的方式,使得在Web浏览器中运行性能密集型应用程序成为可能。这里我们将深入探讨如何编译和测试Wasm文件,以及相关的最佳实践和工具。1.WebAssembly简介: WebAssembly(Wasm)是一......
  • build-essential的作用
    1、build-essential的作用linux操作系统上面开发程序,光有了gcc是不行的它还需要一个build-essential软件包作用是提供编译程序必须软件包的列表信息也就是说编译程序有了这个软件包它才知道头文件在哪才知道库函数在哪还会下载依赖的软件包最后才组成一个开发环境当......
  • [转][SQL Server] 如果想动态 In 应该怎么写?
    来自:TONGYILingma 在SQLServer中,如果你需要动态生成 IN 子句来匹配一组不确定数量的值,你不能直接在预编译的SQL语句中使用一个动态的列表。但你可以通过以下几种方法实现动态IN查询:方法一:参数化查询(适用于少量可变参数)对于少量(通常数据库驱动程序允许的最大......
  • 连接MySQL报错,is not allowed to connect to this MySQL server
    问题描述:        本机装的MySQL数据库,本机可以正常连接,其他机器访问报错,isnotallowedtoconnecttothisMySQLserver,防火墙等其他策略均配置没问题。  解决方案:    出现该问题的原因是,MySQL数据库只允许自身所在的本机器连接,不允许远程连接。1、......
  • ubuntu server安装机房环境以及摄像头部署
    PS:客户安装环境是ubuntu的16.04.5LTS,一般客户都是centos7左右,所以单独写一篇文章本文所需的账号也是root,切换root账号是su需要所需附件的可以打赏后联系我查看ubuntu的版本传输文件到把文件传到/home/sk下安装node:cd/home/sk/tooltar-xvfnode-v8.11.3-lin......
  • java对ServerSocket的开启和关闭
    在通过tcp对接数据的时候,使用java创建tcp服务端来接收客户端的信息处理数据时发现的问题和解决办法1.服务端虽然可以连接多个客户端,缺只能处理第一个客户端的信息2.服务端关闭时,客户端依然可以发送数据解决办法1.因为当第一个客户端连接之后,服务端会一直读取数据造成阻塞,......
  • Android 11 SystemServer启动流程
    在Android11Zygote启动流程有提到,Zygote通过forkSystemServer,fork出SystemServer进程,并在SystemServer进程中调用handleSystemServerProcess返回一个Runnable //...... /*Forchildprocess*/if(pid==0){if(hasSecondZygote(abiList))......
  • jumserver-master版本 lina组件启动报错
    node-vv16.15.1 npm-v8.11.0yarn-v1.22.22 yarnserveyarnrunv1.22.22$vue-cli-serviceserveINFOStartingdevelopmentserver...10%building2/2modules0activeERRORSyntaxError:Cannotuseimportstatementoutsideamodule/opt/lina-ma......