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

告别轮询!用Spring Boot + Vue.js + OkHttp 3.x 轻松搞定SSE实时消息推送(附完整代码)

构建高可靠SSE实时推送系统Spring Boot与Vue.js全栈实践当我们需要在Web应用中实现实时数据推送时传统的轮询方式早已显得力不从心。Server-Sent Events (SSE)技术提供了一种轻量级的解决方案特别适合需要服务器向客户端单向推送数据的场景。本文将带你深入探索如何基于Spring Boot、Vue.js和OkHttp构建一个完整的SSE实时消息系统涵盖从后端实现到前端集成的全流程。1. SSE技术核心原理与优势SSE是一种基于HTTP的服务器推送技术它允许服务器通过持久化的HTTP连接向客户端发送事件流。与WebSocket不同SSE是单向通信服务器到客户端这使得它在实现上更加简单同时具备以下显著优势低延迟相比轮询SSE在消息到达时立即推送给客户端自动重连内置的重新连接机制在网络中断恢复后自动恢复连接简单协议基于纯文本的事件流格式易于调试和实现HTTP兼容无需特殊协议或端口能穿透大多数防火墙SSE与WebSocket对比特性SSEWebSocket通信方向单向(服务器→客户端)双向协议HTTP独立协议连接管理自动重连需手动实现数据格式文本二进制/文本浏览器支持除IE外广泛支持广泛支持在实际项目中SSE特别适合以下场景实时通知系统股票行情推送社交媒体动态更新实时日志监控2. Spring Boot后端实现深度解析Spring Boot通过SseEmitter类提供了对SSE的原生支持下面我们将实现一个完整的SSE服务端并处理各种边界情况。2.1 基础SSE控制器实现RestController RequestMapping(/api/sse) public class SseController { private static final Logger logger LoggerFactory.getLogger(SseController.class); private final SetSseEmitter emitters Collections.newSetFromMap(new ConcurrentHashMap()); GetMapping(/subscribe) public SseEmitter subscribe() { SseEmitter emitter new SseEmitter(30 * 60 * 1000L); // 30分钟超时 emitters.add(emitter); emitter.onCompletion(() - { logger.info(SSE连接正常完成); emitters.remove(emitter); }); emitter.onTimeout(() - { logger.warn(SSE连接超时); emitters.remove(emitter); }); emitter.onError((ex) - { logger.error(SSE连接异常, ex); emitters.remove(emitter); }); return emitter; } }2.2 消息广播与连接管理在实际应用中我们需要考虑消息广播、连接健康检查等高级功能Service public class SseService { Autowired private SseController sseController; private final ScheduledExecutorService scheduler Executors.newSingleThreadScheduledExecutor(); PostConstruct public void init() { // 心跳检测每30秒发送一次 scheduler.scheduleAtFixedRate(this::sendHeartbeat, 30, 30, TimeUnit.SECONDS); } public void broadcast(String eventName, Object data) { String eventId UUID.randomUUID().toString(); String jsonData convertToJson(data); sseController.getEmitters().forEach(emitter - { try { emitter.send(SseEmitter.event() .id(eventId) .name(eventName) .data(jsonData)); } catch (Exception e) { emitter.completeWithError(e); } }); } private void sendHeartbeat() { broadcast(heartbeat, Map.of(timestamp, System.currentTimeMillis())); } }2.3 性能优化与错误处理生产环境中需要考虑的性能优化点连接数限制避免单个客户端创建过多连接消息缓冲处理客户端短暂断开时的消息丢失问题背压控制防止消息生产速度超过消费速度GetMapping(/subscribe) public SseEmitter subscribe(HttpServletRequest request) { String clientId request.getHeader(X-Client-ID); if (StringUtils.isEmpty(clientId)) { throw new IllegalArgumentException(缺少客户端标识); } if (connectionCountPerClient(clientId) MAX_CONNECTION_PER_CLIENT) { throw new IllegalStateException(连接数超过限制); } // ...其余实现 }3. Vue.js前端集成最佳实践现代Vue.js应用特别是Vue 3提供了多种方式来处理SSE连接下面我们探讨Composition API下的实现方式。3.1 基础SSE连接组件template div classsse-container h2实时消息/h2 div v-iferror classerror{{ error }}/div ul classmessage-list li v-formsg in messages :keymsg.id [{{ msg.event }}] {{ msg.data }} /li /ul /div /template script setup import { ref, onMounted, onUnmounted } from vue const messages ref([]) const error ref(null) let eventSource null onMounted(() { eventSource new EventSource(/api/sse/subscribe) eventSource.addEventListener(message, (e) { messages.value.push({ id: Date.now(), event: message, data: e.data }) }) eventSource.addEventListener(error, (e) { error.value 连接发生错误将尝试重新连接 }) }) onUnmounted(() { eventSource?.close() }) /script3.2 高级功能实现自定义事件处理// 在setup函数中添加 eventSource.addEventListener(notification, (e) { const data JSON.parse(e.data) showNotification(data.title, data.content) }) eventSource.addEventListener(heartbeat, () { updateConnectionStatus(active) })自动重连策略const reconnectDelay ref(1000) const maxReconnectDelay 60000 function setupReconnection() { let reconnectTimer null eventSource.addEventListener(error, () { if (eventSource.readyState EventSource.CLOSED) { reconnectTimer setTimeout(() { reconnectDelay.value Math.min( reconnectDelay.value * 2, maxReconnectDelay ) initializeConnection() }, reconnectDelay.value) } }) }4. OkHttp客户端实现与优化在某些场景下我们可能需要用Java客户端消费SSE服务OkHttp提供了完善的SSE支持。4.1 基础SSE客户端public class SseClient { private static final OkHttpClient client new OkHttpClient.Builder() .connectTimeout(10, TimeUnit.SECONDS) .readTimeout(0, TimeUnit.SECONDS) // 禁用读取超时 .build(); public void connect(String url) { Request request new Request.Builder() .url(url) .build(); EventSource.Factory factory EventSource.Factory.create(client); factory.newEventSource(request, new EventSourceListener() { Override public void onOpen(EventSource eventSource, Response response) { System.out.println(SSE连接已建立); } Override public void onEvent(EventSource eventSource, String id, String type, String data) { System.out.printf(收到事件: id%s, type%s, data%s%n, id, type, data); } Override public void onFailure(EventSource eventSource, Throwable t, Response response) { System.err.println(SSE连接失败: t.getMessage()); // 实现重连逻辑 } }); } }4.2 生产级客户端实现对于生产环境我们需要考虑以下增强功能连接池管理重试策略消息处理管道public class RobustSseClient { private final EventSource eventSource; private final ScheduledExecutorService executor; private long reconnectDelay 1000; public RobustSseClient(String url, EventHandler handler) { this.executor Executors.newSingleThreadScheduledExecutor(); OkHttpClient client new OkHttpClient.Builder() .connectionPool(new ConnectionPool(5, 5, TimeUnit.MINUTES)) .build(); Request request new Request.Builder() .url(url) .addHeader(Cache-Control, no-cache) .addHeader(Accept, text/event-stream) .build(); this.eventSource new EventSource.Factory(client) .newEventSource(request, new EventSourceListener() { // 事件处理方法 }); } private void scheduleReconnect() { executor.schedule(() - { reconnectDelay Math.min(reconnectDelay * 2, 60000); connect(); }, reconnectDelay, TimeUnit.MILLISECONDS); } }5. 安全增强与生产部署将SSE服务部署到生产环境时需要考虑以下安全措施认证与授权确保只有合法客户端可以连接速率限制防止滥用CORS配置正确配置跨域策略Spring Security集成示例Configuration EnableWebSecurity public class SecurityConfig extends WebSecurityConfigurerAdapter { Override protected void configure(HttpSecurity http) throws Exception { http .cors().and() .authorizeRequests() .antMatchers(/api/sse/**).authenticated() .and() .addFilterBefore(new SseAuthFilter(), UsernamePasswordAuthenticationFilter.class); } } public class SseAuthFilter extends OncePerRequestFilter { Override protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) { String token request.getHeader(X-SSE-Token); // 验证token逻辑 } }Nginx配置优化server { location /api/sse { proxy_pass http://backend; proxy_http_version 1.1; proxy_set_header Connection ; proxy_set_header X-Client-ID $remote_addr; proxy_buffering off; proxy_read_timeout 24h; } }6. 性能监控与调试技巧完善的监控系统对SSE服务至关重要我们需要跟踪以下指标活跃连接数消息吞吐量错误率Spring Boot Actuator集成Configuration public class SseMetricsConfig { Bean public MeterRegistryCustomizerMeterRegistry sseMetrics() { return registry - { Gauge.builder(sse.connections, SseController.getInstance(), c - c.getActiveConnections()) .description(当前活跃SSE连接数) .register(registry); }; } }客户端调试技巧// 在浏览器控制台调试SSE连接 const es new EventSource(/api/sse/subscribe); es.addEventListener(message, console.log); es.addEventListener(error, console.error);对于复杂问题可以使用Wireshark或tcpdump分析原始网络流量验证事件流格式是否符合规范GET /api/sse/subscribe HTTP/1.1 Accept: text/event-stream Cache-Control: no-cache HTTP/1.1 200 OK Content-Type: text/event-stream Transfer-Encoding: chunked event: message id: 12345 data: {type:notification,content:Hello} event: heartbeat id: 12346 data: 1625097600000

相关文章:

告别轮询!用Spring Boot + Vue.js + OkHttp 3.x 轻松搞定SSE实时消息推送(附完整代码)

构建高可靠SSE实时推送系统:Spring Boot与Vue.js全栈实践 当我们需要在Web应用中实现实时数据推送时,传统的轮询方式早已显得力不从心。Server-Sent Events (SSE)技术提供了一种轻量级的解决方案,特别适合需要服务器向客户端单向推送数据的场…...

基因组模型中的上下文学习与跨模态分析技术

1. 基因组模型与上下文学习的交叉领域探索当生物信息学遇上机器学习的前沿领域,基因组模型中的上下文学习正在打开一扇全新的大门。作为一名在计算生物学领域深耕多年的研究者,我见证了传统基因组分析方法与新兴AI技术的碰撞与融合。这项研究最吸引我的地…...

告别VSCode C++调试噩梦:从‘g++ build active file’报错到一键顺畅调试的避坑全记录

从零构建VSCode C调试环境:一次彻底解决配置问题的实践指南 引言 作为一名长期使用VSCode进行C开发的工程师,我深知配置调试环境时的痛苦。那种看着红色错误提示却无从下手的挫败感,那种在Stack Overflow上翻遍各种解决方案却依然无法解决问题…...

VOIPAC iMX8M开发套件Yocto系统构建与烧录指南

1. VOIPAC iMX8M工业开发套件开箱与初步体验上个月我收到了VOIPAC iMX8M工业级开发套件,这是一款基于NXP i.MX 8M处理器的嵌入式开发平台。开箱时,板卡给我留下了深刻印象——丰富的接口和扩展头让它显得非常灵活。套件预装了Yocto 3.1 Linux系统&#x…...

《计算机学习必看!9 本硬核技术书籍,从入门到进阶全覆盖》

复制文中口令,打开百度app即可接受分享 《摄影测量入门必看!零基础也能看懂的核心基础教程》 口令:怆忾庄单哇它伟荥畔社滔迷弋 《Web 安全入门|万字笔记整理,新手也能直接上手》 口令:怆忾任盾羚曷显工百佚氧盾弋 《编程入门…...

利用多模型能力为内容生成平台提供多样化风格输出

利用多模型能力为内容生成平台提供多样化风格输出 1. 内容生成平台的模型需求挑战 现代内容生成平台需要满足用户对不同文体和风格的多样化需求。从技术角度看,这要求平台能够灵活调用不同特性的大语言模型。传统方案需要为每个模型单独维护API密钥、计费系统和错…...

GitHub 功能全览:涵盖 AI 代码创作、开发者工作流等多领域

导航菜单与外观设置可进行导航切换、登录、外观设置等操作。平台功能包含 AI 代码创作、开发者工作流、应用程序安全、探索等方面。AI 代码创作有 GitHub Copilot、GitHub Spark、GitHub Models、MCP Registry(新);开发者工作流涉及 Actions、…...

开源机器人仪表盘架构设计:从数据采集到Web可视化全链路实践

1. 项目概述:一个面向开源机器人项目的仪表盘最近在折腾一个开源机器人项目,叫 OpenClaw。这名字听起来挺酷,像是某种机械爪或者自动化设备。项目本身在 GitHub 上,由 yusenthebot 这个账号维护。我关注它,是因为想找一…...

Go命令行进度条库bprogress:原理、集成与高级应用指南

1. 项目概述:一个为命令行界面注入活力的进度条工具如果你经常在终端里跑一些耗时较长的任务,比如编译大型项目、批量处理文件,或者下载数据,看着光标在那里一闪一闪,心里是不是总有点没底?不知道任务跑了百…...

Python多线程里用async报错?三步搞定‘There is no current event loop in thread‘

Python多线程中异步编程的陷阱与解决方案:彻底解决"No current event loop"错误 当你在多线程环境中尝试运行异步代码时,那个令人头疼的RuntimeError消息——"There is no current event loop in thread Thread-2"——可能已经让你在…...

从传感器到警报:手把手教你用GEC6818和PWM蜂鸣器搭建环境监控原型(含驱动加载指南)

从传感器到警报:GEC6818开发板与PWM蜂鸣器的环境监控实战指南 当环境温度超过阈值时自动触发警报,这种看似简单的功能背后,是嵌入式系统开发中传感器数据采集、驱动加载和硬件控制的完美结合。本文将带你用GEC6818开发板和PWM蜂鸣器&#xff…...

避坑指南:Flink 使用 Hive 方言时常见的 5 个错误与解决方案(基于 1.13 版本)

Flink与Hive方言深度整合:5个实战避坑指南与解决方案 1. 环境配置与方言切换的常见陷阱 在Flink 1.13版本中,Hive方言的集成已经相对成熟,但配置环节仍然存在几个关键注意点。首先需要明确的是,Hive方言功能必须在HiveCatalog环境…...

30个Illustrator自动化脚本:终极设计效率提升指南

30个Illustrator自动化脚本:终极设计效率提升指南 【免费下载链接】illustrator-scripts Adobe Illustrator scripts 项目地址: https://gitcode.com/gh_mirrors/il/illustrator-scripts 你是否曾在Adobe Illustrator中花费数小时重复同样的操作?…...

金融领域大语言模型应用与可信度评估实践

1. 金融领域大语言模型的应用现状 过去两年,生成式AI技术正在深刻改变金融行业的服务模式。我在某股份制银行AI实验室参与的风控系统升级项目中,首次将大语言模型引入信贷审批流程。实际落地过程中发现,虽然模型在自动化文档处理方面表现出色…...

抄了正点原子的LAN8720原理图,为什么我的板子就是ping不通?分享我的踩坑实录

LAN8720电路设计避坑指南:从原理图复刻到实战排错全解析 第一次看到自己设计的板子卡在0x7809状态码时,那种挫败感至今记忆犹新。作为嵌入式开发者,我们都经历过"照着成熟开发板画原理图却无法工作"的尴尬时刻。本文将用3000字详细…...

AltStore保姆级教程:从下载IPA到成功安装TikTok修改版,避开‘邮件插件’失败坑

AltStore实战指南:无需越狱安装第三方IPA的完整方案 每次iOS系统更新,总有一批实用工具从App Store消失。当官方渠道无法满足需求时,AltStore提供了一种合法合规的解决方案。不同于传统越狱,它利用苹果开发者计划中的免费签名机制…...

本地化多模态RAG桌面应用VectorDB-Plugin:从环境配置到高级调优全指南

1. 项目概述:一个本地化的多模态RAG桌面应用如果你正在寻找一个能彻底摆脱网络依赖、在本地电脑上就能构建个人知识库,并且能处理文档、图片甚至音频文件的智能助手,那么你找对地方了。今天要聊的这个项目,VectorDB-Plugin&#x…...

从实体电池到数字资产——小哈智电用科技承载10年官方回购承诺

两轮锂电行业长久以来困于一个沉默的惯性:产品售出即终结,厂商与电池之间的联系就此断裂。当电池逐渐老化、性能衰减,用户独自承担残值归零的损失,却缺乏可信任的退出通道。“官方回购”之所以在行业内近乎空白,根源并…...

别再只用JSON了!用Apache Avro在Hadoop/Hudi里存数据,性能和空间都赢了

为什么Apache Avro是大数据存储的终极选择?性能与空间的双重胜利 当你在Hadoop或Hudi生态系统中处理PB级数据时,JSON可能正在悄悄消耗你的集群资源和工程师耐心。每次数据扫描时那些冗余的字段名、缓慢的解析速度、膨胀的存储体积——这些都在提醒我们是…...

别再乱接电源了!EP4CE10E22C8N的VCCINT、VCCIO、VCCA引脚供电详解与实战避坑

EP4CE10E22C8N电源系统深度解析:从理论到实践的完整供电方案 当我在实验室第一次调试EP4CE10E22C8N开发板时,遇到了一个令人困惑的现象——FPGA能够正常加载程序,但运行一段时间后随机出现逻辑错误。经过72小时的排查,最终发现问题…...

CSS如何兼容CSS网格区域命名_通过line-based定位实现兼容

grid-area命名区域在IE及旧版Safari/Firefox中完全不支持,应改用line-based定位(如grid-row: 1 / 3),显式声明四值、避免span语法、注意行号从1开始,并为IE11单独提供-ms-grid方案。grid-area命名在旧版浏览器里根本不…...

C# WinForm开发避坑指南:从窗体属性设置到事件处理的5个常见误区与最佳实践

C# WinForm开发避坑指南:从窗体属性设置到事件处理的5个常见误区与最佳实践 在Windows桌面应用开发领域,C# WinForm凭借其成熟的组件库和可视化设计能力,依然是许多开发者的首选。然而,随着项目复杂度提升,不少开发者会…...

NVIDIA Spectrum-X:AI优化的网络平台技术解析

1. NVIDIA Spectrum-X:专为AI优化的网络平台解析NVIDIA Spectrum-X正在迅速成为超大规模云基础设施中AI工作负载的首选网络平台。作为一名长期跟踪数据中心网络技术发展的从业者,我亲眼见证了传统以太网在应对现代AI工作负载时面临的挑战。Spectrum-X的出…...

FPGA程序掉电不丢的秘诀:深入解读MCS文件里的地址与校验码(以Xilinx Flash烧录为例)

FPGA程序掉电不丢的秘诀:深入解读MCS文件里的地址与校验码(以Xilinx Flash烧录为例) 在FPGA开发中,最让人头疼的莫过于调试好的程序在断电后"消失"。不同于传统处理器,FPGA的配置存储器需要特殊的文件格式来…...

人机协同中的三律与反三律

在人机协同与智能系统的构建中,“三律”与“反三律”是一套极具辩证思维的策略框架。简单来说,“三律”是系统内部确保逻辑严谨的“自保之盾”,而“反三律”则是对外博弈时迷惑对手、争取主动的“攻敌之矛”。这一框架通常被应用于“人机环境…...

batch(1) command

文章目录1.简介2.格式3.选项4.示例参考文献1.简介 batch 命令的主要用途是提交一个后台任务。 与 at 命令不同的地方在于 batch 不需要指定时间,自动在系统空闲时执行你交给它的任务。系统空闲指的是系统负载平均值低于 0.8 或 atd 调用中指定的值。 batch 使用方…...

Unity集成OpenAI API实战:GPT对话、DALL·E绘图与Whisper语音全解析

1. 项目概述:在Unity中集成OpenAI的完整方案 如果你正在为你的Unity游戏或应用寻找一种智能对话、内容生成甚至是语音识别的能力,那么将OpenAI的API直接集成到引擎内部,无疑是一条高效且强大的路径。今天要聊的这个 srcnalt/OpenAI-Unity …...

用Python复现经典论文:2006年ALNS算法解决带时间窗的取送货问题(附完整代码)

用Python复现经典ALNS算法:从理论到PDPTW实战 2006年Stefan Ropke提出的自适应大邻域搜索(ALNS)算法,至今仍是解决带时间窗取送货问题(PDPTW)的黄金标准。本文将带您穿越17年技术演进,用现代Python工具链完整复现这一经典算法,并分…...

别再为JSON解析报错头疼了!Jackson的JsonReadFeature帮你搞定13种非标准数据

用Jackson的JsonReadFeature驯服13种非标准JSON数据 当你在深夜调试接口时,突然收到一个JSON解析异常——可能是前端传了个带注释的配置,或是老旧系统吐出了单引号的字符串。这种场景下,Jackson的JsonReadFeature就像瑞士军刀,能帮…...

终极Java面试教程学习环境搭建:5步快速上手Java-Interview-Tutorial

终极Java面试教程学习环境搭建:5步快速上手Java-Interview-Tutorial 【免费下载链接】Java-Interview-Tutorial 请star,勿fork,因为爱force push!涵盖国际大厂Java/数据库/DDD/设计模式/微服务/中间件/AI大模型应用/区块链开发最佳…...