给AI对话加「停止生成」按钮:abort SSE实战

「停止生成」这四个字的按钮,看着简单,我做的时候连踩三个坑。用户点一下要立刻停,还得停得干净------别让请求在后台偷偷跑完烧 token,别让停完之后迟到的数据又冒出来。记一篇实操,全是血泪。

AbortController 是底座

fetch 的中断靠 AbortController,把它的 signal 传进去,调 abort() 就能掐断连接:

javascript 复制代码
const ctrl = new AbortController();

async function send(prompt: string) {
  const res = await fetch('/api/chat', {
    method: 'POST',
    body: JSON.stringify({ prompt }),
    signal: ctrl.signal, // 关键
  });
  const reader = res.body!.getReader();
  // ...读流
}

// 按钮点击
function onStop() {
  ctrl.abort();
}

注意 ctrl 必须每轮对话新建一个------AbortController 是一次性的,abort 过的实例再用,下一次请求一发出去就立刻被中止。我第一版把它提到组件外当单例,结果点过一次停止后,后面所有消息都发不出去,自己还纳闷半天。

坑一:读流的循环要会处理 abort 抛错

abort() 之后,正在 await reader.read() 的那次会抛 AbortError。不 catch 的话控制台一片飘红,看着像崩了。得把它当正常流程吞掉:

javascript 复制代码
try {
  while (true) {
    const { done, value } = await reader.read();
    if (done) break;
    handleChunk(decode(value));
  }
} catch (e) {
  if ((e as Error).name === 'AbortError') {
    // 用户主动停的,不是错误,静默收尾
  } else {
    throw e;
  }
}

AbortError 和真错误分开,是这块代码的体面所在。否则用户每点一次停止,监控就报一条 error,误报能把你淹了。

坑二:React 里 controller 存哪

存 state 会有闭包陷阱------按钮拿到的可能是旧的 controller。用 useRef 存当前这轮的实例才稳:

ini 复制代码
const ctrlRef = useRef<AbortController | null>(null);

async function send(prompt: string) {
  ctrlRef.current?.abort(); // 顺手把上一轮没停的也掐了
  const ctrl = new AbortController();
  ctrlRef.current = ctrl;
  // fetch with ctrl.signal
}

const onStop = () => ctrlRef.current?.abort();

顺带一个体验细节:用户在 AI 还在吐字时又发了条新消息,本质上也该停掉上一轮,所以 send 开头先 abort 一次,一举两得。

坑三:前端 abort,后端不一定停

这个最隐蔽。AbortController 断的只是浏览器到服务端的连接,后端那个调大模型的请求未必跟着停------很多实现里它会闷头把整段生成跑完,token 照烧。我抓包发现点了停止之后,后端日志里那次 completion 还在续,账单不会骗人。

解决得靠后端配合:服务端检测到客户端连接断开(Node 里是 req.on('close'))时,把对上游模型的请求也 abort 掉。前端这边我额外补了个 keepalive 的打点请求,显式告诉后端「这轮我不要了」,双保险。纯前端是兜不住这块成本的,别指望一个 abort 解决所有问题。

php 复制代码
const onStop = () => {
  ctrlRef.current?.abort();
  // 显式通知后端释放上游
  fetch('/api/chat/cancel', { method: 'POST', body: JSON.stringify({ id }), keepalive: true });
};

我用的模型服务本身在 MaaS 那层就支持取消上游生成,省了我自己去管算力释放------前端发个信号,后面停不停干净是平台的事。这种「现成 API + 平台兜底」的活,讯飞这类 MaaS 帮我把后端那截脏活担了,我专注前端这点交互。

你们的停止按钮,后端真的停了吗?建议抓个包看看账单,评论区聊聊有没有人也踩过这个隐形坑。

相关推荐
新新技术迷1 小时前
移动端H5接AI对话的坑:键盘顶起与滚动到底
人工智能
aqi004 小时前
15天学会AI应用开发(七)有了大模型为什么还要引入RAG
人工智能·python·大模型·ai编程·ai应用
用户5191495848455 小时前
libcurl Headers API 释放后重利用漏洞:跨请求复用头句柄导致堆内存安全风险
人工智能·aigc
踩蚂蚁5 小时前
自定义语音唤醒词:从训练到部署的完整链路实践
人工智能
用户5191495848455 小时前
CVE-2025-1094 PostgreSQL SQL注入与WebSocket劫持远程代码执行利用工具
人工智能·aigc
IT_陈寒6 小时前
SpringBoot自动配置这个坑,我踩进去又爬出来了
前端·人工智能·后端
冬奇Lab18 小时前
Agent 系列(23):Web Agent——让 Agent 真正浏览网页
人工智能·llm·agent