FetchAPI 请求流式数据 基本用法

请求 & 请求取消

src\api\fetch.ts

javascript 复制代码
// 创建一个 AbortController 实例,用于取消请求
const controller = new AbortController();
const signal = controller.signal;

// 封装整个请求
//     url:要请求的接口
//     content:要发送的问题参数
//     onMessageReceived:流式数据获取成功后的回调函数
//     end:流式数据获取结束后的回调函数
export const fetchStreamData = async (
  url: any,
  content: any,
  onMessageReceived: any,
  end: any
) => {
  try {
    // 发起fetch请求
    const response = await fetch(url, {
      method: 'POST',
      headers: {
        'Content-Type': 'application/json',
      },
      body: JSON.stringify(content),
      signal,
    });

    const reader = response.body.getReader(); // 流式数据读取器
    let decoder = new TextDecoder(); // 解码器

    // 数据
    let buffer = '';

    while (1) {
      // 异步读取
      const { done, value } = await reader.read();
      if (done) {
        //全部获取流式数据结束后的操作
        end();
        break;
      }

      // 将读取到的数据添加到缓冲区(可能是因为客户端处理数据的速度比服务器发送数据的速度快,
      //导致一次性接收到多条数据。这时候就要用buffer缓冲区来处理一下)
      //后端给我的数据格式:{"chunk_message": "我是数据"}

      // 流式处理‌:当数据以分块形式传输(如网络流、文件分块读取等),每个数据块可能包含字符的一部分字节。
      // stream设置为 true 时,解码器会保留未完成字符的字节,与下一块数据合并解码,避免乱码。 ‌
      // ‌stream 默认值为 false,适用于非流式数据(如完整文件或一次性接收的数据)。
      buffer += decoder.decode(value, { stream: true });

      // 尝试解析每条消息
      let startIndex = buffer.indexOf('{');
      let endIndex = buffer.indexOf('}', startIndex);

      while (startIndex !== -1 && endIndex !== -1) {
        const resultData = buffer.substring(startIndex, endIndex + 1);
        buffer = buffer.substring(endIndex + 1); // 移除已处理的数据

        // 寻找下一条消息的起始位置
        startIndex = buffer.indexOf('{');
        endIndex = buffer.indexOf('}', startIndex);

        // 定义正则表达式来匹配 chunk_message 的值,我们只要chunk_message对应的值
        //因为json数据和text/event-stream数据不同,json数据可以通过.访问chunk_message 的值,
        //但是text/event-stream数据是个纯文本,不能通过.访问chunk_message,我们用正则表达式来获取需要的数据
        let pattern = /"chunk_message"\s*:\s*"([^"]*)"/;

        // 使用正则表达式进行匹配
        let match = resultData.match(pattern);

        if (match) {
          // 提取匹配到的值
          let chunkMsg = match[1];
          // 将消息传递给回调函数处理
          onMessageReceived(chunkMsg);
        } else {
          // console.log("No chunk_message found");
        }
      }
    }
  } catch (error) {
    if (error.name === 'AbortError') {
      console.error('请求取消');
    } else {
      console.error('Fetch error:', error);
    }
  }
};
// 取消请求
export const abortedFetch = () => {
  controller.abort();
};
相关推荐
放下华子我只抽RuiKe58 小时前
React 从入门到生产(四):自定义 Hook
前端·javascript·人工智能·深度学习·react.js·自然语言处理·前端框架
XinZong9 小时前
OpenClaw 实现双重心跳(Heartbeat)+ clawreach虾聊项目实现
javascript
还有多久拿退休金11 小时前
一张栈的图,治好你面试答不出 script 阻塞的病
前端·javascript
zithern_juejin11 小时前
原型与原型链
javascript
008爬虫实战录13 小时前
【码上爬】 题十二:如来神掌 困难, JSVMP加密,使用代理补环境
前端·javascript·node.js
threelab14 小时前
Three.js 数学函数着色器 | 三维可视化 / AI 提示词
javascript·人工智能·着色器
ZC跨境爬虫15 小时前
跟着 MDN 学CSS day_3:(为一个传记页面添加样式)
前端·javascript·css·ui·音视频·html5
夜雪闻竹15 小时前
sql.js WASM 实战:浏览器里跑 SQLite
javascript·sql·wasm
爱喝铁观音的谷力景辉15 小时前
在Cesium中实现带箭头方向路线样式的技术详解
javascript·cesium
Qhappy15 小时前
AI逆向实战:从零还原某航空App的AES加密
javascript·后端