0

0

如何在异步回调场景下将后端处理结果实时推送至前端网页

花韻仙語

花韻仙語

发布时间:2026-01-03 10:04:50

|

685人浏览过

|

来源于php中文网

原创

如何在异步回调场景下将后端处理结果实时推送至前端网页

本文介绍在“客户端→服务a→camunda→服务b→服务a”异步流程中,当服务a收到来自camunda的回调结果后,如何高效、可靠地将该结果传递给原始发起请求的web前端javascript),重点分析轮询、websocket及事件驱动方案的适用性与实现要点。

在典型的异步工作流系统(如 Camunda)集成场景中,前端发起请求后,服务A立即返回“已接收”(如 HTTP 202 Accepted),后续处理由 Camunda 编排并调用服务B完成;待服务B返回结果,Camunda 再通过回调 API 将结果 POST 到服务A的指定 endpoint。此时,服务A需将该结果精准、低延迟地送达最初触发请求的那个浏览器会话——这正是核心挑战:HTTP 是无状态、单向请求-响应模型,而服务A无法主动“推”数据给前端。

以下是三种主流可行方案,按推荐度排序:

✅ 方案一:WebSocket(推荐用于实时性要求高、用户会话明确的场景)

尽管你担心“只用一次就断开”的资源浪费,但 WebSocket 实际开销极小,且现代框架(如 Spring WebFlux + STOMP、Node.js + Socket.IO)支持连接复用、自动心跳与会话绑定。关键在于关联请求上下文

  1. 前端发起请求时,携带唯一 requestId(如 UUID),并同时建立 WebSocket 连接,将 requestId 作为连接标识(例如通过 query 参数或登录态关联);
  2. 服务A在收到 Camunda 回调时,根据回调 payload 中的业务ID(或关联的 requestId)查找到对应 WebSocket Session,并推送结果;
  3. 前端监听对应 requestId 的消息,收到即渲染,随后可主动关闭连接。
// 前端示例(使用原生 WebSocket)
const requestId = 'a1b2c3d4-5678-90ef-ghij-klmnopqrstuv';
const ws = new WebSocket(`wss://api.example.com/ws?rid=${requestId}`);

ws.onmessage = (e) => {
  const data = JSON.parse(e.data);
  if (data.requestId === requestId && data.status === 'completed') {
    document.getElementById('result').innerText = data.payload;
    ws.close(); // 一次性任务,安全关闭
  }
};
⚠️ 注意事项:需在服务端维护 requestId → WebSocketSession 映射(建议使用内存缓存如 Caffeine 或 Redis,设置 TTL 防止内存泄漏);生产环境务必启用 TLS(WSS)和 Origin 校验。

? 方案二:短轮询(Polling)——简单可靠,适合低频/容忍秒级延迟场景

若架构受限(如 CDN 不支持 WebSocket、老旧浏览器兼容需求),可采用轻量级轮询:前端在收到初始 202 响应后,定时 GET /api/status/{requestId} 查询结果。

立即学习前端免费学习笔记(深入)”;

MedPeer
MedPeer

AI驱动的一站式科研服务平台

下载
// 前端轮询逻辑(带退避与超时)
async function pollResult(requestId, maxAttempts = 30, baseDelay = 1000) {
  for (let i = 0; i < maxAttempts; i++) {
    try {
      const res = await fetch(`/api/status/${requestId}`);
      const { status, result } = await res.json();
      if (status === 'completed') {
        renderResult(result);
        return;
      }
      await new Promise(r => setTimeout(r, Math.min(baseDelay * (2 ** i), 10000))); // 指数退避
    } catch (e) {
      console.warn('Poll failed:', e);
      break;
    }
  }
  alert('Request timeout or failed');
}

✅ 优势:零新协议依赖,天然兼容所有环境;
❗ 缺点:增加服务端查询压力(尤其并发高时),存在延迟(最小为轮询间隔)。

? 方案三:Server-Sent Events(SSE)——折中选择,单向推送更轻量

若仅需服务端→前端单向通知(无需双向通信),SSE 比 WebSocket 更简洁:基于 HTTP 长连接,自动重连,浏览器原生支持(EventSource)。

服务端(Spring Boot 示例):

@GetMapping(value = "/events/{requestId}", produces = MediaType.TEXT_EVENT_STREAM_VALUE)
public SseEmitter handleSse(@PathVariable String requestId) {
  SseEmitter emitter = new SseEmitter(30_000L); // 30s 超时
  sseService.registerEmitter(requestId, emitter); // 绑定到 requestId
  return emitter;
}
// 收到 Camunda 回调后:
sseService.sendResult(requestId, resultPayload);

前端:

const eventSource = new EventSource(`/api/events/${requestId}`);
eventSource.onmessage = e => {
  document.getElementById('result').innerText = e.data;
  eventSource.close();
};

? 总结:

  • 首选 WebSocket:语义清晰、实时性强、可控性高,一次连接成本远低于长期轮询;
  • 次选 SSE:适合纯推送场景,比 WebSocket 更轻,但不支持跨域 cookie 认证(需 token 透传);
  • 保底轮询:开发最快,运维最稳,但需谨慎设计重试策略与服务端缓存。
    无论哪种方案,统一使用 requestId 作为全链路追踪与会话绑定的核心标识,是确保结果精准送达的关键设计原则。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

544

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

372

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

728

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

470

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

393

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

990

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

655

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

545

2023.09.20

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

150

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 3.2万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 1.9万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号