当前位置: 首页 > news >正文

实现对deepseek流式返回的json数据,进行逐字解析并实时渲染

要实现对 流式返回的 JSON 数据(如 DeepSeek 的 SSE 流式响应)进行 逐字解析并实时渲染,而不是等待整个 JSON 完整加载后再处理,需要明确以下目标和问题:


文章目录

    • 核心目标
    • 技术方案概述
      • 1. 使用 `fetch` + `ReadableStream` 接收流式数据
      • 2. 读取 chunk 并拼接到 buffer 中
    • 关键难点:如何解析未完成的 JSON?
      • 方法一:**按行或分隔符切分 chunks**
      • 方法二:**构建 JSON 增量解析器**
    • 实时渲染建议
    • 示例:整合完整流程
    • 注意事项

核心目标

  • 逐步接收数据流(chunk by chunk)
  • 逐步拼接字符串
  • 逐步解析 JSON(部分解析)
  • 实时渲染/更新 UI

技术方案概述

1. 使用 fetch + ReadableStream 接收流式数据

const response = await fetch('https://api.deepseek.com/stream-endpoint', {method: 'POST',headers: {'Content-Type': 'application/json'},body: JSON.stringify({prompt: '讲一个故事'})
});const reader = response.body.getReader();
const decoder = new TextDecoder();
let buffer = '';

2. 读取 chunk 并拼接到 buffer 中

while (true) {const { done, value } = await reader.read();if (done) break;buffer += decoder.decode(value, { stream: true });// 尝试解析 JSON 或提取部分内容processBuffer(buffer);
}

关键难点:如何解析未完成的 JSON?

JSON 是结构化的文本格式,如果只收到一部分内容(比如 { "text": "Hello" } 只收到 { "text": "Hel),直接使用 JSON.parse() 会报错。

方法一:按行或分隔符切分 chunks

假设 DeepSeek 返回的是 多段 JSON 对象(每行一个完整 JSON),可以按 \n 分割:

function processBuffer(buffer) {const lines = buffer.split('\n');lines.forEach(line => {if (line.trim() === '') return;try {const json = JSON.parse(line);console.log(json.text); // 实时获取 text 字段updateUI(json.text);    // 实时渲染到页面buffer = '';             // 清空已处理部分} catch (e) {// JSON 不完整,保留 buffer 继续拼接}});
}

方法二:构建 JSON 增量解析器

如果你的数据是一个完整的 JSON(例如 {"content": "Hel..."} 但被分片传输),可以尝试使用增量解析库,如:

  • jsonparse(Node.js / 浏览器可用)

示例:

import sax from 'jsonparse';const parser = new sax.Parser();parser.onValue = function (value) {if (this.stack.length === 1 && this.key === 'content') {updateUI(value); // 实时渲染 content 内容}
};function processBuffer(buffer) {try {parser.write(buffer);} catch (e) {// 忽略不完整 JSON 错误,继续等待更多数据}
}

实时渲染建议

你可以将每次解析出的内容追加到 DOM 中,模拟“打字效果”:

function updateUI(text) {const container = document.getElementById('output');container.textContent += text;container.scrollTop = container.scrollHeight; // 滚动到底部
}

示例:整合完整流程

async function startStreaming() {const response = await fetch('https://api.deepseek.com/stream-endpoint', {method: 'POST',headers: {'Content-Type': 'application/json'},body: JSON.stringify({ prompt: '讲一个故事' })});const reader = response.body.getReader();const decoder = new TextDecoder();let buffer = '';const parser = new sax.Parser();parser.onValue = function (value) {if (this.stack.length === 1 && this.key === 'content') {updateUI(value);}};while (true) {const { done, value } = await reader.read();if (done) break;buffer += decoder.decode(value, { stream: true });try {parser.write(buffer);buffer = ''; // 成功解析后清空 buffer} catch (e) {// JSON 不完整,保留 buffer 等待下一次拼接}}parser.close();
}function updateUI(text) {const container = document.getElementById('output');container.textContent += text;
}

注意事项

项目建议
后端支持确保接口返回的是 text/event-stream 类型,并开启 CORS
错误处理添加网络中断、JSON 解析失败等异常捕获机制
性能优化避免频繁操作 DOM,可使用 requestAnimationFrame 或节流
安全性如果涉及用户输入,需做 XSS 过滤
完整性校验最终仍需在 done 后对最终结果做一次完整 JSON 解析

文章转载自:

http://uzxQsDP6.kphyL.cn
http://cPVKbKGn.kphyL.cn
http://sVIzQDoa.kphyL.cn
http://ThvMCYPj.kphyL.cn
http://r24VKuWZ.kphyL.cn
http://63KcnB4U.kphyL.cn
http://K39JJTHi.kphyL.cn
http://jgM8kekp.kphyL.cn
http://WGJKbaKn.kphyL.cn
http://DzfpTkCq.kphyL.cn
http://JgvgwVzJ.kphyL.cn
http://4Gixuhvw.kphyL.cn
http://sXH3AJWr.kphyL.cn
http://E8p76GJl.kphyL.cn
http://95QzLVMu.kphyL.cn
http://91DQ6mTl.kphyL.cn
http://diudyoyt.kphyL.cn
http://ldUmpmyX.kphyL.cn
http://mg4pR4KV.kphyL.cn
http://IaZMtlm0.kphyL.cn
http://1idx0raq.kphyL.cn
http://WJhvv06A.kphyL.cn
http://v2Kzufla.kphyL.cn
http://Ydw3lw0A.kphyL.cn
http://ppjQnIn1.kphyL.cn
http://W8gjOZS9.kphyL.cn
http://FnEWrbTP.kphyL.cn
http://dRZx8DCQ.kphyL.cn
http://cQ6TGl45.kphyL.cn
http://FxxTZ0fC.kphyL.cn
http://www.dtcms.com/a/228420.html

相关文章:

  • Python中os模块详解
  • 蓝桥杯 k倍区间
  • [蓝桥杯]生物芯片
  • 负载均衡相关基本概念
  • 通过阿里云 DashScope API 调用通义千问
  • [蓝桥杯]求解台阶问题
  • Redis 缓存问题及其解决方案
  • DrissionPage 异常处理实战指南:构建稳健的网页自动化防线
  • Eureka 高可用集群搭建实战:服务注册与发现的底层原理与避坑指南
  • n8n 自动化平台 Docker 部署教程(附 PostgreSQL 与更新指南)
  • (13)java+ selenium->元素定位大法之By_partial_link_text
  • 04 APP 自动化- Appium toast 元素定位列表滑动
  • 内网横向之RDP缓存利用
  • Redis 缓存粒度如何控制?缓存整个对象还是部分字段?
  • Mac 芯片系列 安装cocoapod 教程
  • 【linux】VNC无头显示器启动方法
  • Neo4j 备份与恢复:原理、技术与最佳实践
  • 高效易用的 MAC 版 SVN 客户端:macSvn 使用体验
  • HALCON 深度学习训练 3D 图像的几种方式优缺点
  • 2022年 国内税务年鉴PDF电子版Excel
  • C#基础语法(2)
  • lsinitramfs命令
  • 定时器:中央对齐模式剖析
  • 06-排序
  • Java Lambda 表达式的缺点和替代方案
  • 破局与进阶:ueBIM 在国产 BIM 赛道的差距认知与创新实践
  • 【计算机网络】数据链路层——ARP协议
  • Windows【基础操作2】
  • 【WPF】从普通 ItemsControl 到支持筛选的 ItemsControl:深入掌握 CollectionViewSource 用法
  • 基于深度学习的糖尿病视网膜病变图像分类系统的设计与实现 -完整代码+数据