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

js fetch流式请求 AI动态生成文本,实现逐字生成渲染效果

  1. 开启流式请求:向后端接口发起普通的 fetch,它会返回一个包含 ReadableStream 的 Response 对象
  2. 获取流式读取器:调用 response.body.getReader() 获取一个 ReadableStreamDefaultReader 实例
  3. 循环读取数据块:在 while(true) 循环或 for await 中,通过 reader.read() 或 for await (const chunk of response.body.values()) 拿到 Uint8Array 块
  4. 解码并追加显示:使用 TextDecoder 将二进制数据解码成字符串,然后每获取一段就更新到页面上,无需等待完整返回
    MDN Web Docs

发起流式

const response = await fetch('/api/chat', {method: 'POST',headers: { 'Content-Type': 'application/json' },body: JSON.stringify({ prompt: '你好,AI。' })
});
// response.body 即为 ReadableStream

fetch 默认支持流式响应,response.body 就是一个可读流

获取并使用 Reader

const reader = response.body.getReader();  // 锁定流,获取 reader 实例
const decoder = new TextDecoder('utf-8'); // 用于将 Uint8Array 解码为字符串
let done = false;while (!done) {const { value, done: streamDone } = await reader.read();done = streamDone;if (value) {const chunkText = decoder.decode(value, { stream: true });// 这里拿到了一段字符串 chunkTextappendToPage(chunkText);}
}

reader.read() 每次返回一个包含 { value: Uint8Array, done: boolean } 的 Promise
传入 { stream: true } 可以确保多次调用 decode 时不会丢失跨块字符

将数据边读边显示

<div id="chat"></div>
<script>function appendToPage(text) {const chat = document.getElementById('chat');chat.textContent += text;  // 或者用 chat.innerHTML += 转义/格式化后追加}
</script>

每次读取到 chunkText,就调用一次 appendToPage,实时更新 DOM,无需等到 done === true

React 示例

import React, { useState, useEffect } from 'react';function StreamingChat({ prompt }) {const [text, setText] = useState('');useEffect(() => {let cancelled = false;async function fetchStream() {setText('');const res = await fetch('/api/chat', { method: 'POST', body: JSON.stringify({ prompt }) });const reader = res.body.getReader();const decoder = new TextDecoder();let done = false;while (!done && !cancelled) {const { value, done: streamDone } = await reader.read();done = streamDone;if (value) {const chunk = decoder.decode(value, { stream: true });// 追加新内容setText(prev => prev + chunk);}}}fetchStream();return () => { cancelled = true; };}, [prompt]);return <pre style={{ whiteSpace: 'pre-wrap' }}>{text}</pre>;
}export default StreamingChat;

拓展与注意事项

  • 错误处理:在 reader.read() 或 fetch 抛错时,捕获后展示重试选项
  • 性能优化:若数据量巨大,可考虑每累积一定长度再更新一次状态,避免过多重渲染
  • 兼容性:Safari 对流式 API 支持不完全,若需兼容可使用 polyfill 或退回到普通 fetch().then(res => res.text())
  • 流式 JSON:若后端返回的是以换行分隔的 JSON 对象流,可在 decoder.decode 后按 \n 切分并 JSON.parse 逐条处理

文章转载自:
http://ballute.hfstrb.cn
http://adsorbent.hfstrb.cn
http://chemotropic.hfstrb.cn
http://atmometry.hfstrb.cn
http://bluetongue.hfstrb.cn
http://bundook.hfstrb.cn
http://blobberlipped.hfstrb.cn
http://ark.hfstrb.cn
http://batum.hfstrb.cn
http://bioassay.hfstrb.cn
http://anthologist.hfstrb.cn
http://cannonball.hfstrb.cn
http://bioplasma.hfstrb.cn
http://alderfly.hfstrb.cn
http://busker.hfstrb.cn
http://biradial.hfstrb.cn
http://bi.hfstrb.cn
http://betrothed.hfstrb.cn
http://cabomba.hfstrb.cn
http://billycock.hfstrb.cn
http://bruiser.hfstrb.cn
http://amass.hfstrb.cn
http://artistically.hfstrb.cn
http://bookseller.hfstrb.cn
http://bra.hfstrb.cn
http://badger.hfstrb.cn
http://calipers.hfstrb.cn
http://archaeomagnetism.hfstrb.cn
http://carpentaria.hfstrb.cn
http://bistatic.hfstrb.cn
http://www.dtcms.com/a/185764.html

相关文章:

  • 「光域」系列激光测距传感器:以光为尺,重构空间认知边界
  • Python 处理图像并生成 JSONL 元数据文件 - 固定text版本
  • Oracle19c中的全局临时表
  • 使用vite重构vue-cli的vue3项目
  • TDengine 在金融领域的应用
  • 《Head First 设计模式》第一章 - 笔记
  • 【ASR学习笔记】:语音识别领域基本术语
  • web 自动化之 Unittest 四大组件
  • 一个.Net开发的、用于自动化测试Windows应用程序的开源框架
  • Wpf学习片段
  • 从海洋生物找灵感:造个机器人RoboPteropod,它能在水下干啥?
  • 当生产了~/qt-arm/bin/qmake,可以单独编译其他-源码的某个模块,如下,编译/qtmultimedia
  • Hadoop区别
  • ElasticSearch聚合操作案例
  • 基于MCP的智能体架构设计:实现智能体与外部世界的无缝连接
  • 【NextPilot日志移植】ULog
  • 【免费】2005-2018年各省人均财政收支数据
  • HTTPS全解析:从证书签发到TLS握手优化
  • Qt for Android申请允许管理所有文件权限
  • [网络层]网络层设备路由器
  • 使用基于ARM的低功耗微型单板计算机打造智能家居管理系统中枢
  • 如何应对网站被爬虫和采集?综合防护策略与实用方案
  • ZYNQ处理器在发热后功耗增加的原因分析及解决方案
  • PDM采集数字麦克风数据
  • nvm管理node版本
  • 安卓玩机工具-----安卓机型一款很好用的数据备份恢复软件 支持云端备份
  • Qt创建项目
  • vue+threeJS 大理石贴图
  • WEB安全--RCE--webshell bypass2
  • Dagster Pipes系列-2:增强外部脚本与Dagster的交互能力