Spring Boot整合SSE实现消息推送:跨域问题解决与前后端联调实战

news2025/4/21 20:27:33

摘要

本文记录了一次完整的Spring Boot整合Server-Sent Events(SSE)实现实时消息推送的开发过程,重点分析前后端联调时遇到的跨域问题及解决方案。通过@CrossOrigin注解的实际应用案例,帮助开发者快速定位和解决类似问题。


一、项目背景与需求

开发一个实时订单推送系统,需要实现:

  1. 司机端与服务端的持久化连接
  2. 订单信息实时推送
  3. 客户端主动关闭连接

二、技术方案设计

2.1 技术选型

技术组件作用说明
Spring Boot后端服务框架
SSE服务端推送协议
Vue/原生HTML前端消息展示

2.2 系统架构图

 前端页面(localhost:63342)
       │
       ▼
 Spring Boot服务端(localhost:9000)
       │
       ▼
 订单推送系统(业务逻辑)

三、关键代码实现

3.1 前端实现(监听客户端)

<script>
// 建立SSE连接(关键代码)
const source = new EventSource(
  `http://localhost:9000/service-sse-push/connect?userId=1&identity=1`
);

// 消息接收处理
source.addEventListener("message", e => {
  document.getElementById("message").innerHTML += e.data + '<br/>';
});

// 关闭连接时通知服务端
function sourceClose() {
  fetch(`http://localhost:9000/close?userId=1&identity=1`);
  source.close();
}
</script>

3.2 后端核心控制器

@RestController
@CrossOrigin(origins = "http://localhost:63342") // 解决跨域的关键注解
public class SseController {
    
    // 存储所有连接的Map
    public static Map<String, SseEmitter> sseEmitterMap = new HashMap<>();

    /**
     * 建立SSE连接端点
     * @param userId 用户ID
     * @param identity 身份标识
     */
    @GetMapping("/connect")
    public SseEmitter connect(@RequestParam Long userId, 
                             @RequestParam String identity) {
        SseEmitter emitter = new SseEmitter(60_000L);
        String key = SsePrefixUtils.generatorSseKey(userId, identity);
        sseEmitterMap.put(key, emitter);
        return emitter;
    }

    /**
     * 消息推送端点
     * @param pushRequest 推送请求体
     */
    @PostMapping("/push")
    public String push(@RequestBody PushRequest pushRequest) {
        String key = SsePrefixUtils.generatorSseKey(
            pushRequest.getUserId(), 
            pushRequest.getIdentity()
        );
        
        if (sseEmitterMap.containsKey(key)) {
            sseEmitterMap.get(key).send(pushRequest.getContent());
            return "推送成功";
        }
        return "用户未连接";
    }
}

四、遇到的典型问题

4.1 跨域问题现象

在这里插入图片描述

控制台报错

Access to XMLHttpRequest at 'http://localhost:9000/connect' from origin 
'http://localhost:63342' has been blocked by CORS policy

4.2 问题分析

  1. 前端运行在localhost:63342(WebStorm默认端口)
  2. 后端服务运行在localhost:9000
  3. 浏览器安全策略阻止跨域请求

4.3 解决方案

// 在Controller类上添加注解
@CrossOrigin(origins = "http://localhost:63342") 

// 或在方法级别添加
@GetMapping("/connect")
@CrossOrigin(origins = "http://localhost:63342")
public SseEmitter connect(...) {...}

注解作用说明

  • origins:允许指定的源访问
  • methods:允许的HTTP方法(默认所有)
  • allowedHeaders:允许的请求头
  • maxAge:预检请求缓存时间

五、其他优化实践

5.1 连接保活机制

// 添加心跳检测
emitter.onTimeout(() -> {
    log.info("连接超时:{}", key);
    sseEmitterMap.remove(key);
});

emitter.onCompletion(() -> {
    log.info("连接完成:{}", key);
    sseEmitterMap.remove(key);
});

5.2 线程安全改进

// 改用ConcurrentHashMap
public static Map<String, SseEmitter> sseEmitterMap = new ConcurrentHashMap<>();

5.3 连接监控看板

@Scheduled(fixedRate = 5000)
public void printConnections() {
    log.info("当前活跃连接数:{}", sseEmitterMap.size());
}

六、完整联调流程

6.1 启动顺序

  1. 启动Spring Boot服务(端口9000)
  2. 用浏览器打开前端页面(端口63342)

6.2 测试步骤

  1. 点击页面按钮建立连接
  2. 通过Postman发送推送请求
  3. 观察页面消息展示
  4. 点击关闭按钮断开连接

测试结果示例

用户[1_1] 成功建立连接
收到订单消息:新订单到达,请及时处理
用户[1_1] 主动断开连接

七、经验总结

  1. 跨域问题定位:当出现CORS policy错误时,首先检查前端请求地址与后端允许的源
  2. 连接管理要点
    • 及时清理失效连接
    • 使用线程安全集合
    • 添加心跳检测机制
  3. 生产环境建议
    • 使用Nginx反向代理统一端口
    • 配置HTTPS加密通信

踩坑心得

跨域问题就像程序员的"拦路虎",通过这次实践深刻理解了浏览器同源策略的重要性。@CrossOrigin虽好,但生产环境需要更严格的安全控制。

讨论话题
你在使用SSE时还遇到过哪些奇葩问题?欢迎在评论区分享你的踩坑经历!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2318938.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【工具分享】vscode+deepseek的接入与使用

目录 第一章 前言 第二章 获取Deepseek APIKEY 2.1 登录与充值 2.2 创建API key 第三章 vscode接入deepseek并使用 3.1 vscode接入deepseek 3.2 vscode使用deepseek 第一章 前言 deepseek刚出来时有一段时间余额无法充值&#xff0c;导致小编没法给大家发完整的流程&…

康谋方案 | AVM合成数据仿真验证方案

随着自动驾驶技术的快速发展&#xff0c;仿真软件在开发过程中扮演着越来越重要的角色。仿真传感器与环境不仅能够加速算法验证&#xff0c;还能在安全可控的条件下进行复杂场景的重复测试。 本文将分享如何利用自动驾驶仿真软件配置仿真传感器与搭建仿真环境&#xff0c;并对…

Linux内核IPv4路由选择子系统

一、基本知识 1.具体案例&#xff1a;直连路由 结构fib_nh表示下一跳&#xff0c;包含输出网络设备、外出接口索引等信息。 有两个以太网局域网 LAN1 和 LAN2&#xff0c;其中 LAN1 包含子网 192.168.1.0/24&#xff0c;而 LAN2 包含子网 192.168.2.0/24。在这两个 LAN 之…

NWAFU 生物统计实验二 R语言版

#1 setwd(修改为你的工作路径或桌面路径) feed_types <- c("A", "B", "C") weight_gain_means <- c(36.8, 34.9, 21.3) weight_gain_sds <- c(2.4, 2.7, 6.6) weight_gain <- rnorm(3, mean weight_gain_means, sd weight_gain_sd…

Thinkphp指纹识别

识别ThinkPHP框架(指纹) 1.ioc判断 /favicon.ico 2.报错 /1 然后使用工具梭哈

【AVRCP】蓝牙AVRCP协议中的L2CAP互操作性要求深度解析

目录 一、L2CAP互操作性要求&#xff08;针对AVRCP&#xff09; 1.1 核心概念 1.2 AVRCP对L2CAP的增强需求 1.3 关键机制解析 1.4 浏览通道优化配置 1.5 实际应用场景与解决方案 二、通道类型与配置 2.1. 通道类型限制 2.2 PSM字段规范 2.3. 实现意义 3.4. 实际应用…

剑指 Offer II 111. 计算除法

comments: true edit_url: https://github.com/doocs/leetcode/edit/main/lcof2/%E5%89%91%E6%8C%87%20Offer%20II%20111.%20%E8%AE%A1%E7%AE%97%E9%99%A4%E6%B3%95/README.md 剑指 Offer II 111. 计算除法 题目描述 给定一个变量对数组 equations 和一个实数值数组 values 作…

掌握 WRF/Chem 模式:突破大气环境研究技术瓶颈的关键

技术点目录 第一部分、WRF-Chem模式应用案例和理论基础第二部分、Linux环境配置及WRF-CHEM第三部分、WRF-Chem模式编译&#xff0c;排放源制作第四部分、WRF-Chem数据准备&#xff08;气象、排放、初边界条件等&#xff09;&#xff0c;案例实践第五部分、模拟结果提取、数据可…

linux性能监控的分布式集群 prometheus + grafana 监控体系搭建

prometheusgrafana分布式集群资源监控体系搭建 前言一、安装 prometheus二、在要监控的服务器上安装监听器三、prometheus服务器配置四、grafana配置大屏五、创建Linux监控看板五、监控windows服务器注意事项 前言 Prometheus 是一个开源的 ​分布式监控系统 和 ​时间序列数据…

数字化转型 2.0:AI、低代码与智能分析如何重塑企业竞争力?

引言&#xff1a;数字化转型进入2.0时代 在过去的十几年里&#xff0c;企业的数字化转型&#xff08;1.0&#xff09;主要围绕信息化和自动化展开&#xff0c;例如引入ERP、CRM等系统&#xff0c;提高办公效率&#xff0c;减少人为失误。然而&#xff0c;随着市场竞争加剧&…

基于SpringBoot的“校园招聘网站”的设计与实现(源码+数据库+文档+PPT)

基于SpringBoot的“校园招聘网站”的设计与实现&#xff08;源码数据库文档PPT) 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;SpringBoot 工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 系统整体功能图 局部E-R图 系统首页界面 系统注册…

由LAC自动建立L2TP实验

一、实验拓扑: 二、实验配置 1.LAC的配置 基础配置: [LAC]int g 0/0/0 [LAC-GigabitEthernet1/0/0]ip address 192.168.0.1 24 [LAC]int g 1/0/0 [LAC-GigabitEthernet1/0/0]ip address 10.1.1.254 24 [LAC-GigabitEthernet1/0/0]int g1/0/1 [LAC-GigabitEthernet1/0/1]ip ad…

内网渗透(CSMSF) 构建内网代理的全面指南:Cobalt Strike 与 Metasploit Framework 深度解析

目录 1. Cobalt Strike 在什么情况下会构建内网代理&#xff1f; 2. Cobalt Strike 构建内网代理的主要作用和目的是什么&#xff1f; 3. Cobalt Strike 如何构建内网代理&#xff1f;需要什么条件和参数&#xff1f; 条件 步骤 参数 4. Cobalt Strike 内网代理能获取什…

SAP FAGLL03 追加并显示描述字段

目录 1、新建一个结构2、操作FAGLPOSX结构3、新建一个BADI 1、新建一个结构 1.1、先在SE11中新建一个结构&#xff1a;ZZADD_FIELDS_FAGL&#xff0c;把我们要显示的描述字段放在这个结构中 2、操作FAGLPOSX结构 2.1、在FAGLPOSX结构中选择Append Structure&#xff0c;把我…

Linux Vim 寄存器 | 从基础分类到高级应用

注&#xff1a;本文为 “vim 寄存器” 相关文章合辑。 英文引文&#xff0c;机翻未校。 中文引文&#xff0c;略作重排。 未整理去重&#xff0c;如有内容异常&#xff0c;请看原文。 Registers 寄存器 Learning Vim registers is like learning algebra for the first ti…

Ubuntu版免翻墙搭建BatteryHistorian

摘要 昨天安装了一个翻墙版本的很不好用&#xff0c;主要是网络不稳定&#xff0c;故于是换了一个免翻墙的docker镜像。但是发现还是很难用。又安装了一个window版本的免翻墙的BatteryHistorian。明天再分享下Windows的免翻墙的BatteryHistorian步骤。 安装好Docker了就直接d…

Django Rest Framework 创建纯净版Django项目部署DRF

描述创建纯净版的Django项目和 Django Rest Framework 环境的部署 一、创建Django项目 1. 环境说明 操作系统 Windows11python版本 3.9.13Django版本 V4.2.202. 操作步骤(在Pycharm中操作) 创建Python项目drfStudy、虚拟环境 ​虚拟环境中安装 jdangopip install django==4.…

深度洞察:DeepSeek 驱动金融行业智能化转型变革

该文章为软件测评&#xff0c;不是广告&#xff01;&#xff01;&#xff01;&#xff01; 目录 一.金融行业的智能化转型浪潮​ 二.DeepSeek的核心技术剖析 1.DeepSeek 模型的金融智慧​ 2.实时联网搜索&#xff1a;把握金融市场脉搏​ 3.RAG 能力&#xff1a;铸就精准金…

面试题精选《剑指Offer》:JVM类加载机制与Spring设计哲学深度剖析-大厂必考

一、JVM类加载核心机制 &#x1f525; 问题5&#xff1a;类从编译到执行的全链路过程 完整生命周期流程图 关键技术拆解 编译阶段 查看字节码指令&#xff1a;javap -v Robot.class 常量池结构解析&#xff08;CONSTANT_Class_info等&#xff09; 类加载阶段 // 手动加载…

掌握些许 IPv6 要点,windows 远程桌面安全便利两相宜!

掌握这些要点&#xff0c;Windows 远程桌面安全便利两相宜&#xff01; 在日常办公中&#xff0c;许多人会用到 Windows 系统的远程桌面功能。但在实际使用时&#xff0c;会遇到内网计算机难以通过运营商的动态 ip 与多层 NAT 向互联网暴露端口的技术问题&#xff0c;和计算机…