使用 WebSocket 连接 SignalR Hub 时无法接收消息的解决方案

使用 websocket 连接 signalr hub 时无法接收消息的解决方案

本文旨在解决在使用原生 WebSocket 连接 SignalR Hub 时,客户端无法接收到服务端推送消息的问题。文章将分析可能的原因,并提供一种通过修改 SignalR 库的导入方式来解决问题的方案,从而避免使用 SignalR 库时出现的构建错误。

当尝试使用 WebSocket 直接连接 SignalR Hub 时,可能会遇到 onmessage 事件未触发,导致无法接收服务端消息的问题。虽然 onopen 事件能够正常触发,表明 WebSocket 连接已经建立,但消息却无法正确传递。以下提供一种可能的解决方案,以及一些排查思路。

问题分析

SignalR 协议协商: SignalR 使用特定的协议进行通信,并非简单的 WebSocket 协议。在建立连接之前,客户端需要与服务端进行协议协商,获取连接所需的 connectionToken。消息格式: SignalR 使用特定的消息格式(通常是 JSON)进行数据传输。如果客户端发送或接收的消息格式不正确,会导致通信失败。库的导入方式: 使用 Rollup 等打包工具时,不同的导入方式可能会影响库的正确加载和使用。

解决方案

如果遇到 default is not exported by @microsoft/signalr 错误,可能是因为导入方式不正确。尝试以下导入方式:

import * as signalR from "@microsoft/signalr";

而不是:

import signalR, { HubConnection } from "@microsoft/signalr";

这种导入方式可以避免 Rollup 打包时出现的问题,确保 SignalR 库能够正确加载。

详细步骤

协议协商: 使用 fetch API 向 /negotiate 端点发送 POST 请求,获取 connectionToken。

let url = "https://localhost:7143/chatHub";const headers = {};const [name, value] = getUserAgentHeader();headers[name] = value;fetch(`${url}/negotiate?negotiateVersion=1`, {    method: "POST",    headers: {        "Content-Type": "application/json",        headers,    },    body: JSON.stringify({        content: "",        headers: headers,        timeout: 100000,        withCredentials: true,    }),}).then((response) => response.json()).then((response) => {    // The fetch return here some information about signalR connection, and the token for idendification    console.log(response);    // The web socket for SignalR's connection    const wsUrl = `${url}?id=${response.connectionToken}`.replace(/^http/, "ws");    const webSocket = new WebSocket(wsUrl);    webSocket.onopen = (event) => {        // It's triggered        console.log(event);    }    webSocket.onmessage = (message) => {        // It's not triggered        console.log(message);    }    webSocket.onerror = (event) => {        console.log(event);        console.log(event.error);    }    webSocket.onclose = (event) => {        console.log(event);    }});

构建 WebSocket URL: 将 connectionToken 附加到 WebSocket URL。

const wsUrl = `${url}?id=${response.connectionToken}`.replace(/^http/, "ws");

创建 WebSocket 连接: 使用构建好的 URL 创建 WebSocket 连接。

const webSocket = new WebSocket(wsUrl);

处理消息: 确保 onmessage 事件处理函数能够正确解析 SignalR 消息格式。

webSocket.onmessage = (message) => {    // 解析 SignalR 消息格式    console.log(message);}

其他注意事项

跨域问题: 确保服务端允许跨域请求,否则 WebSocket 连接可能会被阻止。防火墙 检查防火墙设置,确保 WebSocket 连接没有被阻止。服务端配置: 确保 SignalR Hub 在服务端正确配置,并且正在发送消息。

总结

通过修改 SignalR 库的导入方式,可以解决在使用 Rollup 等打包工具时可能出现的构建错误。同时,需要确保正确进行协议协商,构建 WebSocket URL,并处理 SignalR 消息格式。如果问题仍然存在,可以检查跨域、防火墙和服务端配置等因素。

以上就是使用 WebSocket 连接 SignalR Hub 时无法接收消息的解决方案的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1539231.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月21日 05:48:05
下一篇 2025年12月21日 05:51:09

相关推荐

  • JavaScript缓存策略_Service Worker离线存储

    Service Worker 是实现离线缓存的核心技术,通过拦截网络请求并结合 Cache API 管理资源,支持缓存优先、网络优先等多种策略;需注册、安装、激活并监听 fetch 事件来控制缓存,版本更新时应清理旧缓存以确保生效。 Service Worker 是实现 JavaScript 缓存和…

    好文分享 2025年12月21日
    000
  • JavaScript时间处理_时区转换与日历组件

    JavaScript的Date对象基于UTC存储时间戳但显示时转为本地时区,处理时区和日历需注意:构造函数解析字符串时区分是否含时区信息,推荐使用toISOString输出UTC时间;跨时区显示可用toLocaleString配合timeZone选项(如America/New_York);构建日历时…

    2025年12月21日
    000
  • JavaScript防抖与节流_性能优化实践

    防抖和节流是前端优化高频事件的两种手段:防抖通过延迟执行并仅响应最后一次操作,适用于搜索输入等场景;节流则保证固定时间间隔内最多执行一次,适合滚动监听等持续响应需求。 在前端开发中,频繁触发的事件(如窗口滚动、输入框输入、鼠标移动)容易导致性能问题。为优化这类场景,防抖(Debounce)和节流(T…

    2025年12月21日
    000
  • JavaScript缓存策略_CDN与边缘计算优化

    通过浏览器缓存、CDN加速与边缘计算协同优化JavaScript加载。1. 浏览器强缓存配合内容哈希实现长期缓存与自动更新;2. CDN将JS分发至全球节点,降低延迟并提升可用性;3. 边缘计算在近用户端执行逻辑,支持动态注入与请求预处理;4. 综合策略包括资源分离、缓存协调与边缘增强,构建高效交付…

    2025年12月21日
    000
  • Angular递归组件实现动态导航菜单:基于配置高效构建复杂UI

    本文详细介绍如何在angular应用中利用递归组件模式,根据动态配置数据生成多级导航菜单。通过定义清晰的数据接口和构建可复用的组件,我们能够高效地渲染任意深度的嵌套菜单,从而提升代码的可维护性和扩展性,并实现基于层级的动态样式和行为控制。 引言 在现代Web应用开发中,动态生成复杂的用户界面元素,尤…

    2025年12月21日
    000
  • 深入理解JavaScript中setTimeout的执行机制与事件循环

    javascript的`settimeout`函数提供了一个最小延迟,但其回调的实际执行时间并非绝对精确。这是因为javascript是单线程的,并且依赖事件循环机制。长时间运行的同步代码会阻塞主线程,导致`settimeout`的回调函数在指定延迟结束后,仍需等待调用栈清空才能被执行。 在Java…

    2025年12月21日
    000
  • JavaScript代码规范_ESLint插件开发

    ESLint插件开发需创建含rules、configs的npm包,如eslint-plugin-myteam;编写规则函数遍历AST节点,例如禁止console.log需监听MemberExpression并用context.report报错;通过RuleTester测试有效与无效代码案例;发布后在…

    2025年12月21日
    000
  • 函数组合:compose与pipe实现方法

    函数组合通过compose(从右到左)和pipe(从左到右)实现,分别使用reduceRight和reduce串联函数,提升代码抽象与可读性。 在函数式编程中,函数组合是一种将多个函数串联起来,前一个函数的输出作为下一个函数输入的技术。常见的两种实现方式是 compose 和 pipe,它们的核心思…

    2025年12月21日
    000
  • JavaScript算法优化_时间复杂度分析与改进

    时间复杂度是衡量算法效率的核心指标,优化可提升性能与体验。1. 理解O(1)、O(n)、O(n²)、O(log n)等表示法,识别高复杂度操作;2. 减少嵌套循环,如用哈希表将两数之和从O(n²)降为O(n);3. 善用Set、Map及内置方法filter、map等提高效率;4. 排序预处理结合双指…

    2025年12月21日
    000
  • JavaScript模块联邦_微前端架构设计

    模块联邦是Webpack 5实现微前端的核心技术,支持运行时动态加载远程子应用模块。主应用通过remotes引用子应用,子应用通过exposes暴露组件,shared配置共享依赖。例如主应用配置remotes引入remoteApp@http://localhost:3001/remoteEntry.…

    2025年12月21日
    000
  • Express中间件怎么自定义_Express中间件原理与自定义开发教程

    中间件是Express中处理请求的核心,通过函数访问req、res和next,可实现日志、权限校验等功能。1. 基本结构为接收req、res、next参数的函数,需调用next()继续流程;2. 使用app.use()全局注册或在路由中局部使用;3. 常见场景包括请求日志记录(如打印方法和路径)、身…

    2025年12月21日
    000
  • JS模块导入怎么实现_JS模块化importexport语法使用指南

    JavaScript模块化通过export和import实现代码拆分与复用,支持命名导出、默认导出及动态导入,需注意文件扩展名、type=”module”设置及服务器运行环境。 JavaScript 模块化让开发者能将代码拆分成多个文件,提高可维护性和复用性。通过 impor…

    2025年12月21日
    000
  • 如何用js脚本制作轮播图_js轮播图效果实现与脚本编写步骤

    轮播图通过HTML结构、CSS样式和JavaScript交互实现图片切换。1. HTML构建容器、图片列表、按钮和指示点;2. CSS设置图片横向排列与过渡效果;3. JS控制索引切换、按钮响应、自动播放及指示点联动;4. 可扩展鼠标悬停暂停、触摸滑动等优化功能。 轮播图是网页中常见的交互组件,用于…

    2025年12月21日 好文分享
    000
  • JS正则表达式常用方法与实践_javascript正则

    JavaScript中正则表达式通过字面量或RegExp构造函数创建,常用修饰符包括g、i、m、u、s;核心方法有test、exec、match、replace和split,广泛应用于表单验证、数据提取与文本替换,如手机号验证、URL参数解析、HTML标签过滤及数字千分位格式化,熟练掌握可高效处理字…

    2025年12月21日
    000
  • JS尾调用优化_递归函数改进方案

    尾调用优化通过消除尾递归的栈帧避免栈溢出,虽JS引擎支持有限,但可改写为循环或结合蹦床实现高效递归。 JavaScript中的尾调用优化(Tail Call Optimization, TCO)是一种提升递归函数性能的技术,尤其在处理深度递归时能有效避免栈溢出问题。虽然部分JS引擎尚未完全支持ES6…

    2025年12月21日
    000
  • JS数组扁平化_多种方法性能对比

    数组扁平化方法包括:1. flat()语法简洁但兼容性差;2. reduce+concat逻辑直观但内存开销大;3. 扩展运算符+循环适合中等深度但性能随层数下降;4. 栈模拟法避免递归,适合深层结构;5. toString+split仅限数字且最快。性能上,小数组用flat(),大数组用栈模拟,纯…

    2025年12月21日
    000
  • js concat连接数组

    concat()方法用于合并数组并返回新数组,原数组不变;可连接多个数组或值,支持“打平”一层嵌套,如arr1.concat(arr2, 5, [6,7])得[1,2,3,4,5,6,7];现代常用扩展运算符替代。 在 JavaScript 中,concat() 方法用于合并两个或多个数组,并返回一…

    2025年12月21日
    000
  • React 应用中点击按钮导致页面刷新的问题分析与解决

    MetaData: Key Value {assetData.metaData.map((item, idx) => ( updateMetaData(e, idx)} /> updateMetaData(e, idx)} /> {assetData.metaData.length…

    2025年12月21日
    000
  • 动态调整列表项顺序的JavaScript实践指南

    本教程详细探讨了如何使用%ignore_a_1%正确地动态调整html列表项的顺序。我们将分析jquery `before()` 方法的常见误用,并提供两种有效的解决方案:一是修正 `before()` 的参数顺序以实现预期效果,二是推荐使用更具语义化和鲁棒性的 `prependto()` 方法将特…

    2025年12月21日
    000
  • Web Components Shadow DOM与外部CSS框架的集成实践

    本文探讨了web components shadow dom的样式隔离特性及其对外部css框架的影响。我们将深入理解shadow dom如何防止样式泄露和冲突,并提供在web component内部有效引入并应用如uikit等外部css框架的实践方法。通过在shadow dom内部重新加载样式表,开…

    2025年12月21日
    000

发表回复

登录后才能评论
关注微信