首页 > 其他分享 >ChatGPT流式数据传输探索

ChatGPT流式数据传输探索

时间:2024-09-20 11:52:01浏览次数:3  
标签:uid sseEmitterMap 流式 连接 sseEmitter SSE 数据传输 ChatGPT logger

文章目录

背景介绍

用过GPT类语言模型的同学都知道,其在返回数据时都是一个字或几个字的显示,你是否思考过它是怎么传输的?经过一番查询学习,了解到了SSE,GPT就是通过SSE流式传输方式进行传输的。
SSE 全称为 Server-sent-events , 是一种基于 HTTP 协议的通信技术,允许服务器主动向客户端(通常是Web浏览器)发送更新。它是 HTML5 标准的一部分,设计初衷是用来建立一个单向的服务器到客户端连接,使得服务器可以实时地向客户端发送数据。这种服务端实时向客户端发送数据的传输方式,其实就是流式传输。今天就来模拟实践一番SSE传输。

目标

实现SSE流式数据传输

技术细节

1、客户端

Server-Sent Events(SSE)是 HTML5 的一部分,用于从服务器实时接收更新,目前大部分主流浏览器都提供了支持:

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>Demo</title>
  </head>

  <body>
    <div id="result"></div>
    <script>
      let result = "";
      if ("EventSource" in window) {
        // 连接服务器
        var sseSource = new EventSource(
          "http://127.0.0.1:8080/test/createSse?uid=123"
        );

        // 连接打开
        sseSource.onopen = function () {
          console.log("连接打开");
          document.getElementById("result").innerText += '连接打开\n';
        };

        // 连接错误
        sseSource.onerror = function (err) {
          console.log("连接错误:", err);
        };
        sseSource.onclose = function () {
          console.log("连接关闭");
        };
        // 接收到数据
        sseSource.onmessage = function (event) {
          console.log("接收到数据:", event.data);
          if (event.data.startsWith("[DONE]")) {
            document.getElementById("result").innerText+='数据传输完毕\n'
          } else {
            handleReceiveData(event);
          }
        };

        // 关闭链接
        function handleCloseSse() {
          console.log("关闭链接");
          sseSource.close();
        }

        // 处理服务器返回的数据
        function handleReceiveData(data) {
          try {
            let json = JSON.parse(data.data);
            document.getElementById("result").innerText += json.message;
          } catch (error) {}
          //
        }
      }
    </script>
  </body>
</html>


2、服务端

我们目前服务端主要使用 SpringBoot,其对 SSE 主要提供了两种支持:

  1. Spring WebMVC:传统的基于 Servlet 的同步阻塞编程模型,即 同步模型Web框架。
  2. Spring WebFlux:异步非阻塞的响应式编程模型,即 异步模型Web框架。

今天使用spring-webmvc,我用的版本是5.2.15
首先在springBoot项目中新建个SseClient.java类,代码如下


public class SseClient {
    private static final Logger logger = LoggerFactory.getLogger(SseClient.class);
    private static final Map<String, SseEmitter> sseEmitterMap = new ConcurrentHashMap<>();
    /**
     * 创建连接
     */
    public SseEmitter createSse(String uid) {
        //默认30秒超时,设置为0L则永不超时
        SseEmitter sseEmitter = new SseEmitter(0l);
        //完成后回调
        sseEmitter.onCompletion(() -> {
            logger.info("[{}]结束连接...................", uid);
            sseEmitterMap.remove(uid);
        });
        //超时回调
        sseEmitter.onTimeout(() -> {
            logger.info("[{}]连接超时...................", uid);
        });
        //异常回调
        sseEmitter.onError(
                throwable -> {
                    try {
                        logger.info("[{}]连接异常,{}", uid, throwable.toString());
                        sseEmitter.send(SseEmitter.event()
                                .id(uid)
                                .name("发生异常!")
                                .data("发生异常请重试!")
                                .reconnectTime(3000));
                        sseEmitterMap.put(uid, sseEmitter);
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
        );
        try {
            sseEmitter.send(SseEmitter.event().reconnectTime(5000));
        } catch (IOException e) {
            e.printStackTrace();
        }
        sseEmitterMap.put(uid, sseEmitter);
        logger.info("[{}]创建sse连接成功!", uid);
        return sseEmitter;
    }

    /**
     * 给指定用户发送消息
     *
     */
    public boolean sendMessage(String uid,String messageId, String message) {
        if (StrUtil.isBlank(message)) {
            logger.info("参数异常,msg为null", uid);
            return false;
        }
        SseEmitter sseEmitter = sseEmitterMap.get(uid);
        if (sseEmitter == null) {
            logger.info("消息推送失败uid:[{}],没有创建连接,请重试。", uid);
            return false;
        }
        try {
            sseEmitter.send(SseEmitter.event().id(messageId).reconnectTime(1*60*1000L).data(message));
            logger.info("用户{},消息id:{},推送成功:{}", uid,messageId, message);
            return true;
        }catch (Exception e) {
            sseEmitterMap.remove(uid);
            logger.info("用户{},消息id:{},推送异常:{}", uid,messageId, e.getMessage());
            sseEmitter.complete();
            return false;
        }
    }


    /**
     * 断开
     * @param uid
     */
    public void closeSse(String uid){
        if (sseEmitterMap.containsKey(uid)) {
            SseEmitter sseEmitter = sseEmitterMap.get(uid);
            sseEmitter.complete();
            sseEmitterMap.remove(uid);
        }else {
            logger.info("用户{} 连接已关闭",uid);
        }

    }
}

接着新建一个Controller,内容如下


@RestController
@RequestMapping(value ="/test")
public class TestController {

    @Autowired
    private SseClient sseClient;
    @CrossOrigin
    @GetMapping("/createSse")
    public SseEmitter createConnect(String uid) {
        return sseClient.createSse(uid);
    }
  }

现在先将后端代码运行起来,然后运行前端代码,效果如下
在这里插入图片描述
在这里插入图片描述
可以看到已经连接成功。
接下来我们来模拟数据发送,在controller里面增加个发送消息的接口

	@CrossOrigin
    @GetMapping("/sendMsg")
    public String sseChat(String uid) {

        for (int i = 0; i < 10; i++) {
            JSONObject jo = new JSONObject();
            jo.put("uid", uid);
            jo.put("message", "服务器消息" + i + "\n");
            sseClient.sendMessage(uid, "no" + i, jo.toString());
            try {
                //睡眠一下,模拟GPT数据处理
                Thread.sleep(500);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        //告诉前端数据发送完毕
        sseClient.sendMessage(uid, "", "[DONE]");
        return "ok";
    }

然后重新启动后,就可以进行消息发送了,这里我用postman来调用发送消息的接口
在这里插入图片描述
在这里插入图片描述
可以看到前端只发送一次请求,服务器端的数据通过事件流的方式传输到前端。

总结

SSE 与 WebSocket 作用相似,都是建立浏览器与服务器之间的通信渠道,然后服务器向浏览器推送信息。
总体来说,WebSocket 更强大和灵活。因为它是全双工通道,可以双向通信;SSE 是单向通道,只能服务器向浏览器发送,因为流信息本质上就是下载。如果浏览器向服务器发送信息,就变成了另一次 HTTP 请求。但是,SSE 也有自己的优点。

  1. SSE使用 HTTP 协议,现有的服务器软件都支持。WebSocket 是一个独立协议。
  2. SSE属于轻量级,使用简单;WebSocket 协议相对复杂。
  3. SSE 默认支持断线重连,WebSocket 需要自己实现。
  4. SSE 一般只用来传送文本,二进制数据需要编码后传送,WebSocket 默认支持传送二进制数据。
  5. SSE 支持自定义发送的消息类型。

因此,两者各有特点,可根据自己的使用场景选择。

标签:uid,sseEmitterMap,流式,连接,sseEmitter,SSE,数据传输,ChatGPT,logger
From: https://blog.csdn.net/xs_2012/article/details/142378253

相关文章

  • 文心一言 VS 讯飞星火 VS chatgpt (350)-- 算法导论24.1 1题
    一、在图24-4上运行Bellman-Ford算法,使用结点作为源结点。在每一遍松弛过程中,以图中相同的次序对每条边进行松弛,给出每遍松弛操作后的值和值。然后,把边的权重改为,再次运行该算法,这次使用作为源结点。如果要写代码,请用go语言。文心一言:首先,我们根据题目要求,手动通过Bellman-Ford算......
  • 大人时代变了,ChatGPT使用指南(喂嘴里)
    目录一、面向软件开发人员的ChatGPT提示词二、AI能力对比和推荐三、AI能做什么国外ChatGPT的大模型工具使用对于国内大部分人来说仍然有比较大的门槛,比如网络访问限制问题,账户注册限制,账户封号等问题。那么在国内,有没有一些可替代工具呢?这篇文章就给大家分享一些高效的......
  • 国内可以使用的ChatGPT镜像网站【9月持续更新】
    首先基础知识还是要介绍得~一、模型知识:GPT-4o:最新的版本模型,支持视觉等多模态,OpenAI文档中已经更新了GPT-4o的介绍:128k上下文,训练截止2023年10月(作为对比,GPT-4-Turbo截止2023年12月)。GPT-4Turbo:支持视觉等多模态,128k上下文,训练截止2023年12月。GPT-3.5......
  • 亲测好用,ChatGPT 3.5/4.0新手使用手册~
     都知道ChatGPT很强大,聊聊天、写论文、搞翻译、写代码、写文案、审合同等等,无所不能~那么到底怎么使用呢?其实很简单了,国内AI产品发展也很快,很多都很好用了~我一直在用,建议收藏下来~  有最先进、最新的GPT模型,还有很多其他效率工具都是在各自领域,绝对领先地位的产品~......
  • 如何使用ChatGPT帮你写论文?有思路有教程【小白上手指南】
    停留5分钟看完这篇文章,绝对让你写论文如虎添翼我这里先给你提供一下思路,再进行详细说明一、框架1、设定ChatGPT的背景你把主题和开题报告以及要求发给它,并告知是什么学位等等,此处建议你做一个专门的助手2、列出大纲要求它列出写作大纲,你根据它写的大纲看是否要进行调整,如......
  • ChatGPT 多媒体应用设计师备考考点讲解(七):音视频编码与传输优化策略
    音视频编码与传输技术是多媒体应用设计中至关重要的环节。高效的音视频编码不仅能够保证内容质量,还可以减少带宽占用,提升传输效率。而在传输过程中,优化音视频流的传输策略可以降低延迟、减少丢包、保证播放的流畅性。在本篇文章中,我们将详细讲解音视频编码的基础知识、常见编码格式......
  • 人工智能 | 基于ChatGPT开发人工智能服务平台
    简介ChatGPT在刚问世的时候,其产品形态就是一个问答机器人。而基于ChatGPT的能力还可以对其做一些二次开发和拓展。比如模拟面试功能、或者智能机器人功能。模拟面试功能包括个性化问题生成、实时反馈、多轮面试模拟、面试报告。智能机器人功能提供24/7客服支持、自然语言处理、任......
  • 文心一言 VS 讯飞星火 VS chatgpt (349)-- 算法导论23.2 8题
    八、Borden教授提出了一个新的分治算法来计算最小生成树。该算法的原理如下:给定图,将划分为两个集合和,使得和的差最多为1。设为端点全部在中的边的集合,为端点全部在中的边的集合。我们递归地解决两个子图和的最小生成树问题。最后,在边集合中选择横跨切割和的最小权重的边来将求出的......
  • 2024.9.1_ChatGPT镜像列表
    收集自网络,更新于2024.9.1ChatGPT国内镜像ChatGPT外国镜像......
  • ChatGPT中Java相关问答(包括Java基础知识和一些面试题)
    分享一个自己学习Java时的记录ChatGPT中的对话:ChatGPT链接包括如下问题HowtolearnJavainordertobecomeasoftwaredevelopmentengineer,pleasegivedetailsofthestudyprogramaswellasthereferencestudymaterials.详细解释java中的this引用解释一下类、......