springboot学习(五十八) springboot中使用SseEmitter推送消息

本文介绍了如何在SpringBoot应用中使用SseEmitter进行服务器到客户端的消息推送。通过创建SseEmitter实例,设置超时和重连时间,实现实时推送。客户端使用EventSource接收消息,支持onmessage、onerror和onopen事件。示例代码展示了订阅、推送和断开连接的处理方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

服务端常用推送技术有:
1、客户端轮询:ajax定时拉取
2、服务端主动推送:WebSocket。全双工的,本质上是一个额外的tcp连接,建立和关闭时握手使用http协议,其他数据传输不使用http协议,更加复杂一些,适用于需要进行复杂双向数据通讯的场景。
3、服务端主动推送:SSE (Server Send Event)。html5新标准,用来从服务端实时推送数据到浏览器端,
直接建立在当前http连接上,本质上是保持一个http长连接,轻量协议简单的服务器数据推送的场景,使用服务器推送事件, SSE技术是基于单工通信模式,只是单纯的客户端向服务端发送请求,服务端不会主动发送给客户端。服务端采取的策略是抓住这个请求不放,等数据更新的时候才返回给客户端,当客户端接收到消息后,再向服务端发送请求,周而复始。

这篇文章介绍Springboot中使用SseEmitter实现消息推送。

1、服务端:

package com.iscas.biz.test.sse;

import org.springframework.http.MediaType;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.servlet.mvc.method.annotation.SseEmitter;

import java.io.IOException;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 *  测试SSE推送消息
 *
 * @author zhuquanwen
 * @vesion 1.0
 * @date 2021/9/11 21:01
 * @since jdk1.8
 */
@RestController
@RequestMapping(path = "sse/test")
public class SseControllerTest {
    private static Map<String, SseEmitter> sseCache = new ConcurrentHashMap<>();
    @GetMapping(path = "subscribe", produces = {MediaType.TEXT_EVENT_STREAM_VALUE})
    public SseEmitter push(String id) throws IOException {
        // 超时时间设置为3s,用于演示客户端自动重连
        SseEmitter sseEmitter = new SseEmitter(30000L);
        // 设置前端的重试时间为1s
        sseEmitter.send(SseEmitter.event().reconnectTime(1000).data("连接成功"));
        sseCache.put(id, sseEmitter);
        System.out.println("add " + id);
        sseEmitter.onTimeout(() -> {
            System.out.println(id + "超时");
            sseCache.remove(id);
        });
        sseEmitter.onCompletion(() -> System.out.println("完成!!!"));
        return sseEmitter;
    }

    @GetMapping(path = "push")
    public String push(String id, String content) throws IOException {
        SseEmitter sseEmitter = sseCache.get(id);
        if (sseEmitter != null) {
            sseEmitter.send(SseEmitter.event().name("msg").data("后端发送消息:" + content));
        }
        return "over";
    }

    @GetMapping(path = "over")
    public String over(String id) {
        SseEmitter sseEmitter = sseCache.get(id);
        if (sseEmitter != null) {
            sseEmitter.complete();
            sseCache.remove(id);
        }
        return "over";
    }
}

其中subscribe是开启并订阅消息,push是模拟触发后端推送,over是模拟断开sse连接。

2、客户端:

<!doctype html>
<html lang="en">
<head>
    <title>Sse测试</title>
    <meta charset="utf-8"/>
</head>
<body>
<div>sse测试</div>
<div id="result"></div>
</body>
</html>
<script>
    var source = new EventSource('http://localhost:7901/demo/sse/test/subscribe?id=qwe');
    //source.onmessage = function (event) {
     //   text = document.getElementById('result').innerText;
      //  text += '\n' + event.data;
       // document.getElementById('result').innerText = text;
    //};
    source.addEventListener("msg", function(e) {
        text = document.getElementById('result').innerText;
        text += '\n' + e.data;
        console.log(e);
        document.getElementById('result').innerText = text;
        //source.close();
    })
    source.onerror = function(e) {
        if (e.readyState == EventSource.CLOSE) {
            text = document.getElementById('result').innerText;
            text += '\n' + "连接关闭";
        }
    }
    <!-- 添加一个开启回调 -->
    source.onopen = function (event) {
        text = document.getElementById('result').innerText;
        text += '\n 开启: ';
        console.log(event);
        document.getElementById('result').innerText = text;
    };
</script>

3、测试
在这里插入图片描述

### 回答1: 这是一个关于使用逗号分隔输入一组数字,然后输出其中最大值的问题。答案是: ``` nums = input("请输入一组数字,用逗号分隔:").split(",") max_num = max(nums) print("最大值为:", max_num) ``` ### 回答2: Python 是一种高级编程语言,用于解决各种问题。在 Python 中,要输入数字组成的列表或元组,可以使用内置的 input() 函数。该函数会通过屏幕提示用户输入,然后返回一个表示输入内容的字符串。 为了能够计算列表中的最大值,我们需要将输入的字符串解析为一个列表。在 Python 中,可以通过使用 split() 方法将字符串按照指定的分隔符分割成多个子字符串,并将这些子字符串放在一个列表中。对于以逗号作为分隔符的数字列表,我们可以使用下面的代码: ``` num_str = input("请输入一组数字(用逗号分隔):") num_list = [int(num) for num in num_str.split(",")] ``` 这段代码首先提示用户输入一组数字,然后调用 split() 方法将这个字符串按照逗号分隔成多个子字符串。接着,使用列表推导式将每个子字符串转换为整数,并将其放在一个新的列表中。 有了列表后,我们可以使用内置的 max() 函数找到列表中的最大值。max() 函数可以接受任意多个参数,并返回这些参数中的最大值。因此,最简单的计算列表最大值的方式是直接将列表中的所有元素传递给 max() 函数,如下所示: ``` max_num = max(num_list) print("最大值为:", max_num) ``` 这段代码调用了 max() 函数,并将计算结果赋值给变量 max_num。接着,使用 print() 函数输出最大值。 综上所述,计算 Python 中输入一组数字,采用逗号分隔,输出其中的最大值的完整代码为: ``` num_str = input("请输入一组数字(用逗号分隔):") num_list = [int(num) for num in num_str.split(",")] max_num = max(num_list) print("最大值为:", max_num) ``` 以上的代码片段能够满足我们所提出的需求,输入数据按照逗号分隔,然后计算并输出最大值。需要注意的是,这段代码仅适用于输入为整数的情况。如果列表中包含浮点数或其他类型的数据,可能需要对代码进行修改。 ### 回答3: 要实现输入一组数字,采用逗号分隔的功能,可以使用Python内置的input()函数。下面通过一个代码示例来演示如何实现这个功能: num_list = input("请输入一组数字,用逗号隔开:").split(",") # 使用input()函数获取用户输入 # 使用split()函数将输入的字符串按逗号分割成一个列表 max_num = float(num_list[0]) # 假设第一个数是最大值 for num in num_list: if float(num) > max_num: max_num = float(num) # 遍历列表中的每个数字 # 如果某个数字比当前最大值大,则更新最大值 print("输入的数字为:", num_list) print("其中最大值为:", max_num) # 输出结果 代码说明: 1. 用户输入一组数字,使用input()函数获取。 2. 使用split()函数将输入的字符串按逗号分隔成一个列表。 3. 假设第一个数是最大值,遍历列表中的每个数字,如果某个数字比当前最大值大,则更新最大值。 4. 输出输入的数字列表和其中的最大值。 值得注意的是,我们使用了float()函数将输入的数字字符串转换成了浮点数,以便于进行比较大小。 最后,这个简单的代码示例提供了一种Python实现输入一组数字,采用逗号分隔,输出其中的最大值的方法。这种方法可以轻松应对类似的问题,让我们的Python编程更加高效。
评论 11
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值