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

Spring Boot实战:5分钟搞定SSE消息推送(含完整代码示例)

Spring Boot实战5分钟构建股票行情推送系统SSE全流程指南1. 为什么选择SSE技术在实时数据推送领域开发者常面临技术选型的困惑。当我们需要实现股票行情更新这类高频单向数据推送场景时Server-Sent EventsSSE往往是最优雅的解决方案。与传统的轮询和复杂的WebSocket相比SSE具备三大核心优势极简实现基于标准HTTP协议无需额外协议栈自动恢复内置连接中断重试机制资源高效单个连接支持持续数据流// 传统轮询 vs SSE 请求对比 GetMapping(/polling) // 传统轮询 public String getStockPrice() { return stockService.getLatestPrice(); // 每次请求都需建立新连接 } GetMapping(/sse) // SSE public SseEmitter streamStockPrice() { return sseService.createEmitter(); // 单次连接持续推送 }提示SSE特别适合股票行情、新闻推送、实时监控等场景这些场景中90%以上的数据流是服务器到客户端的单向传输2. 快速搭建SSE服务端2.1 基础环境配置确保Spring Boot项目包含web依赖Spring Boot 2.7或3.0dependency groupIdorg.springframework.boot/groupId artifactIdspring-boot-starter-web/artifactId /dependency2.2 核心服务层实现创建StockSSEService.java处理连接管理Service Slf4j public class StockSSEService { private final MapString, SseEmitter emitters new ConcurrentHashMap(); public SseEmitter subscribe(String stockCode) { SseEmitter emitter new SseEmitter(30 * 60 * 1000L); // 30分钟超时 emitter.onCompletion(() - { log.info({} SSE连接正常关闭, stockCode); emitters.remove(stockCode); }); emitter.onTimeout(() - { log.warn({} SSE连接超时, stockCode); emitters.remove(stockCode); }); emitters.put(stockCode, emitter); return emitter; } public void pushPriceUpdate(String stockCode, String price) { if (emitters.containsKey(stockCode)) { try { emitters.get(stockCode).send( SseEmitter.event() .id(UUID.randomUUID().toString()) .data(price) ); } catch (IOException e) { log.error({} 推送失败: {}, stockCode, e.getMessage()); emitters.remove(stockCode); } } } }2.3 控制器层设计StockController.java暴露两个关键端点RestController RequestMapping(/api/stocks) public class StockController { Autowired private StockSSEService sseService; GetMapping(/subscribe/{code}) public SseEmitter subscribe(PathVariable String code) { return sseService.subscribe(code); } PostMapping(/mock-update/{code}) public ResponseEntity? mockUpdate( PathVariable String code, RequestParam String price) { sseService.pushPriceUpdate(code, price); return ResponseEntity.ok().build(); } }3. 前端实时监听实现3.1 基础事件监听div idstock-ticker h3股票代码: span idstock-code/span/h3 div idprice-display--/div /div script const stockCode AAPL; // 示例股票代码 document.getElementById(stock-code).textContent stockCode; const eventSource new EventSource(/api/stocks/subscribe/${stockCode}); eventSource.onmessage (event) { document.getElementById(price-display).textContent $${parseFloat(event.data).toFixed(2)}; flashPriceChange(); // 价格变化视觉反馈 }; eventSource.onerror () { console.error(SSE连接异常); // 实现自动重连逻辑... }; /script3.2 增强型处理带事件类型eventSource.addEventListener(priceUpdate, (e) { updatePrice(e.data); }); eventSource.addEventListener(volumeAlert, (e) { showAlert(交易量异常: ${e.data}); });4. 生产环境进阶配置4.1 性能优化参数配置项推荐值说明connectionTimeout180000030分钟连接超时毫秒heartbeatInterval1500015秒心跳间隔防止代理超时maxConnectionsPerClient3每个客户端最大连接数限制Configuration public class SSEConfig implements WebMvcConfigurer { Override public void configureAsyncSupport(AsyncSupportConfigurer configurer) { configurer.setDefaultTimeout(1800000); configurer.setTaskExecutor(taskExecutor()); } Bean public ThreadPoolTaskExecutor taskExecutor() { ThreadPoolTaskExecutor executor new ThreadPoolTaskExecutor(); executor.setCorePoolSize(10); executor.setMaxPoolSize(50); executor.setQueueCapacity(100); return executor; } }4.2 安全防护措施连接验证GetMapping(/subscribe/{code}) public SseEmitter subscribe( PathVariable String code, RequestHeader(Authorization) String token) { if (!securityService.validateToken(token)) { throw new SecurityException(无效凭证); } return sseService.subscribe(code); }CORS配置Bean public WebMvcConfigurer corsConfigurer() { return new WebMvcConfigurer() { Override public void addCorsMappings(CorsRegistry registry) { registry.addMapping(/api/stocks/**) .allowedOrigins(https://yourdomain.com) .allowedMethods(GET); } }; }5. 典型问题排查指南5.1 常见问题速查表现象可能原因解决方案连接立即断开响应头缺失确保返回text/event-stream数据接收延迟代理缓冲设置X-Accel-Buffering: no浏览器限制连接数同源连接数限制使用HTTP/2或多子域名长时间无数据心跳缺失定期发送注释行:保持连接5.2 心跳机制实现Scheduled(fixedRate 15000) public void sendHeartbeat() { emitters.forEach((code, emitter) - { try { emitter.send(SseEmitter.event().comment(hb)); } catch (IOException e) { emitter.completeWithError(e); } }); }6. 扩展应用场景6.1 多股票订阅方案public SseEmitter subscribeMultiple(RequestParam ListString codes) { SseEmitter emitter new SseEmitter(); codes.forEach(code - { StockWatcher watcher new StockWatcher(code, price - { try { emitter.send( SseEmitter.event() .name(code) .data(price) ); } catch (IOException ex) { // 错误处理 } }); stockService.registerWatcher(watcher); }); return emitter; }6.2 与消息队列集成KafkaListener(topics stock-updates) public void handleStockUpdate(StockUpdate update) { if (emitters.containsKey(update.getCode())) { emitters.get(update.getCode()) .send(update.toSSEEvent()); } }7. 性能压测数据参考使用JMeter对SSE服务进行压力测试单台4核8G服务器并发连接数平均响应时间吞吐量内存占用10023ms4250/sec1.2GB50067ms3800/sec2.8GB1000142ms2900/sec4.5GB关键优化建议使用HTTP/2减少连接开销对非活跃连接实施清理策略考虑分布式部署时使用Redis发布/订阅8. 完整代码示例8.1 服务端增强版// 在StockSSEService中添加历史数据推送 public SseEmitter subscribeWithHistory(String code) { SseEmitter emitter subscribe(code); executor.execute(() - { try { // 推送最近10条历史数据 stockService.getHistoryPrices(code, 10) .forEach(price - { emitter.send(price.toSSEEvent()); }); } catch (Exception e) { emitter.completeWithError(e); } }); return emitter; }8.2 前端增强版// 添加重连逻辑 function setupSSE() { const eventSource new EventSource(url); eventSource.onerror () { eventSource.close(); setTimeout(setupSSE, 3000); // 3秒后重连 }; return eventSource; } // 添加数据缓存 const priceHistory []; eventSource.onmessage (e) { priceHistory.push({ time: new Date(), price: e.data }); renderChart(priceHistory); };

相关文章:

Spring Boot实战:5分钟搞定SSE消息推送(含完整代码示例)

Spring Boot实战:5分钟构建股票行情推送系统(SSE全流程指南) 1. 为什么选择SSE技术? 在实时数据推送领域,开发者常面临技术选型的困惑。当我们需要实现股票行情更新这类高频单向数据推送场景时,Server-Sent…...

Stable Yogi Leather-Dress-Collection 实战案例:为智能车内饰提供皮革设计方案

Stable Yogi Leather-Dress-Collection 实战案例:为智能车内饰提供皮革设计方案 最近几年,智能车这个概念越来越火。大家讨论的焦点,往往集中在自动驾驶、智能座舱、车机系统这些“硬核”科技上。但作为一个和设计、材料打过不少交道的人&am…...

UOS Server 20下MLNX_OFED驱动编译踩坑实录:从fput缺失到成功安装的全过程

UOS Server 20下MLNX_OFED驱动编译实战:从内核兼容性到模块修复的深度解析 在国产操作系统生态快速发展的今天,UOS Server 20作为企业级Linux发行版,正逐步获得更多行业用户的青睐。然而,当我们需要在UOS上部署高性能网络设备时&a…...

如何为你的应用选择靠谱的IP归属地数据源?一份给开发者的选型指南

在开发需要显示用户所在地的功能时,一个准确、稳定的数据服务是底层支撑。无论是展示用户属地,还是电商与内容平台的区域化运营,都依赖于此。然而,市面上的数据源质量参差不齐,有的更新不及时导致新分配的地址无法识别…...

别再只会点灯了!用STM32CubeMX配置外部中断控制电机启停(附完整代码)

从GPIO到电机控制:STM32CubeMX外部中断实战指南 在嵌入式开发中,GPIO点灯往往是初学者的第一个实验,但真正的工程应用远不止于此。想象一下工业场景中的紧急停止按钮——当操作员拍下急停开关时,系统必须立即停止所有电机运转&…...

谷歌账号安全提示终极指南:为什么关闭插件就能登录?底层机制解析

谷歌账号安全机制深度解析:插件权限与登录拦截的底层逻辑 每次遇到谷歌账号登录被拦截的提示,大多数用户的第一反应是"换个浏览器试试"。但很少有人追问:为什么关闭插件就能解决问题?这背后涉及一套复杂的安全评估体系。…...

SE(3)-Transformers实战:如何用等变注意力网络处理3D点云数据(附PyTorch代码)

SE(3)-Transformers实战:3D点云处理的等变注意力网络全解析 在3D计算机视觉领域,点云数据因其无序性和非结构化特点,一直是深度学习处理的难点。传统卷积神经网络在处理这类数据时面临诸多挑战,而等变神经网络的出现为这一领域带来…...

LabelImg+YOLOv8:零基础打造专属目标检测模型(附完整数据集配置模板)

LabelImgYOLOv8:零基础打造专属目标检测模型(附完整数据集配置模板) 目标检测技术正在重塑各行各业的智能化进程,从工业质检到智慧零售,从医疗影像到自动驾驶,这项技术让机器真正拥有了"看懂世界"…...

如何在Windows上快速将OpenModelica模型转为FMU并导入Simulink(2023最新版)

2023年Windows平台OpenModelica转FMU全流程指南:零依赖方案与Simulink无缝集成 在工程仿真领域,多工具协同已成为提升工作效率的关键策略。对于使用OpenModelica进行物理建模却需要在Simulink中完成控制算法验证的工程师而言,FMU(…...

Unity铰链四杆机构仿真:从机械原理到代码实现的保姆级教程

Unity铰链四杆机构仿真:从机械原理到代码实现的保姆级教程 在游戏开发和工业仿真领域,机械结构的动态模拟一直是个既有趣又具挑战性的课题。铰链四杆机构作为机械传动的基础构件,其运动轨迹的精确模拟能为游戏中的机关设计、机器人动画乃至工…...

EVA-02企业内网部署方案:基于内网穿透的安全访问实践

EVA-02企业内网部署方案:基于内网穿透的安全访问实践 最近和几个做企业AI应用的朋友聊天,发现大家有个共同的痛点:想用EVA-02这类强大的视觉模型,但又担心直接把服务暴露在公网上有安全风险。公司内部的数据、代码,哪…...

Dify Rerank性能翻倍实操手册:从BM25到Cross-Encoder,3类重排序算法在真实QA场景中的Latency/Recall/MRR对比数据全公开

第一章:Dify Rerank性能翻倍实操手册:从BM25到Cross-Encoder,3类重排序算法在真实QA场景中的Latency/Recall/MRR对比数据全公开在真实企业级问答系统中,重排序(Reranking)是决定最终检索质量的关键环节。我…...

面试必问的Saga模式:从补偿事务设计到高频考点解析(附避坑指南)

分布式事务Saga模式:面试高频考点与实战避坑指南 在当今微服务架构盛行的时代,分布式事务处理已成为开发者必须掌握的核心技能之一。Saga模式作为解决分布式事务问题的经典方案,因其优雅的设计理念和良好的扩展性,在技术面试中频繁…...

Ostrakon-VL-8B与JavaScript前端交互:构建实时图片分析Web应用

Ostrakon-VL-8B与JavaScript前端交互:构建实时图片分析Web应用 你有没有想过,让网站不仅能显示图片,还能“看懂”图片?比如,用户上传一张照片,网站立刻就能告诉你照片里有什么、是什么风格,甚至…...

Ubuntu 22.04实时核编译避坑指南:解决NVIDIA驱动安装常见问题

Ubuntu 22.04实时核编译与NVIDIA驱动深度避坑指南 如果你正在Ubuntu 22.04上尝试编译实时内核并安装NVIDIA驱动,那么这篇文章就是为你准备的。作为一位经历过无数次内核编译和驱动安装的老手,我深知这个过程有多么令人抓狂——从依赖项缺失到内核模块签名…...

Ricon组态系统集成指南

Ricon组态系统完整技术文档 基于Web的可视化组态编辑和实时监控平台 版本:v1.0.1 | 更新时间:2024年12月 文档目录 系统概述技术架构核心功能组件系统开发指南最佳实践故障排查 1. 系统概述 1.1 产品简介 Ricon组态系统是一个基于Web技术的可视化组态…...

Youtu-VL-4B-Instruct WebUI源码部署手册:含端口映射、HTTPS配置、跨域访问解决方案

Youtu-VL-4B-Instruct WebUI源码部署手册:含端口映射、HTTPS配置、跨域访问解决方案 1. 引言:为什么选择Youtu-VL-4B-Instruct? 如果你正在寻找一个既能看懂图片又能和你聊天的AI助手,那么腾讯优图实验室开源的Youtu-VL-4B-Inst…...

Golang GORM 零值更新实战:从 Struct 到 Map 的解决方案

1. 为什么GORM会忽略零值更新? 这个问题困扰过不少刚接触GORM的开发者。想象一下这样的场景:你在开发一个学生成绩管理系统,需要将某个学生的分数从100分调整为0分。按照常规思路,你会把结构体中的Score字段设为0,然后…...

Pikachu靶场实战:PHP反序列化漏洞代码审计与利用

1. PHP反序列化漏洞基础入门 第一次接触PHP反序列化漏洞时,我和大多数新手一样感到困惑:为什么把字符串转换成对象就能产生安全风险?这得从PHP的序列化机制说起。记得去年我在审计一个CMS系统时,就因为忽略了这个小功能导致整个系…...

云容笔谈国风IP孵化:从单张人像生成到虚拟偶像全生命周期管理方案

云容笔谈国风IP孵化:从单张人像生成到虚拟偶像全生命周期管理方案 1. 东方美学影像创作新纪元 在数字内容创作蓬勃发展的今天,如何将传统文化精髓与现代技术完美融合,成为许多创作者面临的挑战。传统的人像生成工具往往难以准确把握东方美学…...

芯片设计必备:Synopsys ICC中的时钟树综合(CTS)优化技巧与实战解析

芯片设计必备:Synopsys ICC中的时钟树综合(CTS)优化技巧与实战解析 时钟树综合(Clock Tree Synthesis, CTS)是芯片物理实现流程中的关键环节,其质量直接影响芯片的时序收敛和功耗表现。在先进工艺节点下,时钟网络的复杂性和重要性愈发凸显。本…...

PyTorch广播机制详解:为什么你的张量运算突然报错?

PyTorch广播机制详解:为什么你的张量运算突然报错? 在深度学习项目中,张量运算的维度匹配问题就像编程中的"指针错误"一样令人头疼。当你信心满满地运行一个看似简单的矩阵乘法时,突然跳出的RuntimeError: The size of …...

Qwen3-Reranker-0.6B实战:用vLLM和Gradio搭建重排序服务

Qwen3-Reranker-0.6B实战:用vLLM和Gradio搭建重排序服务 1. 引言:为什么需要重排序服务 在信息检索和问答系统中,重排序(Re-ranking)是提升结果质量的关键环节。想象一下,当你在搜索引擎输入一个问题时&a…...

使用EasyExcel实现多Sheet数据导出与Web端下载的完整指南

1. 为什么选择EasyExcel处理Excel数据 在Java生态中处理Excel文件,很多开发者第一时间会想到Apache POI。这个老牌工具确实功能强大,但我在实际项目中发现,当处理超过10万行数据时,POI经常会出现内存溢出(OOM&#xff…...

SECS/GEM协议实战:用Python解析半导体设备通信的二进制数据流

SECS/GEM协议实战:用Python解析半导体设备通信的二进制数据流 半导体制造设备的通信协议SECS/GEM是工业自动化领域的核心标准之一。不同于常见的文本协议,SECS/GEM中大量使用二进制数据流传输设备状态、工艺参数等关键信息。本文将聚焦实际开发中最棘手的…...

Qwen2-VL-2B-Instruct效果展示:Transformer架构下的多模态理解惊艳案例

Qwen2-VL-2B-Instruct效果展示:Transformer架构下的多模态理解惊艳案例 最近在尝试各种多模态模型,一个绕不开的话题就是如何在有限的资源下,获得足够好的图文理解能力。很多大模型效果好,但对硬件要求也高,部署起来总…...

轻量化对决:nanobot镜像vs原版OpenClaw资源占用实测

轻量化对决:nanobot镜像vs原版OpenClaw资源占用实测 1. 测试背景与动机 最近在折腾本地AI助手时,我发现OpenClaw虽然功能强大,但资源占用一直是个痛点。特别是当我想在老旧笔记本上跑起来时,经常遇到内存不足的问题。正好看到社…...

STM32程序烧录成功却“跑飞”?从启动到外设的深度排障指南

1. 硬件配置问题排查 当你遇到STM32程序烧录成功但运行异常时,硬件问题往往是首要排查对象。我遇到过太多因为硬件配置不当导致的"灵异事件",有些问题甚至会让开发者怀疑人生。下面我们就从最基础的硬件配置开始,一步步揭开这些问题…...

Mermaid Live Editor:文本驱动的图表创作革新

Mermaid Live Editor:文本驱动的图表创作革新 【免费下载链接】mermaid-live-editor Edit, preview and share mermaid charts/diagrams. New implementation of the live editor. 项目地址: https://gitcode.com/GitHub_Trending/me/mermaid-live-editor 价…...

Z-Image-Turbo_Sugar脸部Lora一键部署教程:基于Python入门的环境配置指南

Z-Image-Turbo_Sugar脸部Lora一键部署教程:基于Python入门的环境配置指南 你是不是也刷到过那些风格独特、一眼就能认出来的AI人像?比如那种带着甜美糖系风格,五官精致又有点梦幻感的头像。以前总觉得做出这种效果需要很高的技术门槛&#x…...