前端WebSocket实时通信:别再用轮询了!

前端WebSocket实时通信:别再用轮询了!

毒舌时刻

WebSocket?听起来就像是前端工程师为了显得自己很专业而特意搞的一套复杂技术。你以为随便用个WebSocket就能实现实时通信?别做梦了!到时候你会发现,WebSocket连接断开的问题让你崩溃,重连机制让你晕头转向。

你以为WebSocket是万能的?别天真了!WebSocket在某些网络环境下会被防火墙拦截,而且服务器的负载也是个问题。还有那些所谓的WebSocket库,看起来高大上,用起来却各种问题。

为什么你需要这个

  1. 实时性:WebSocket提供全双工通信,可以实现真正的实时通信,比轮询更高效。
  2. 减少网络流量:WebSocket只需要建立一次连接,减少了HTTP请求的开销。
  3. 服务器推送:服务器可以主动向客户端推送数据,而不需要客户端轮询。
  4. 低延迟:WebSocket的延迟比轮询低,适合实时应用。
  5. 更好的用户体验:实时通信可以提供更好的用户体验,比如实时聊天、实时数据更新等。

反面教材

// 1. 简单WebSocket连接 const socket = new WebSocket('ws://localhost:8080'); socket.onopen = function(event) { console.log('WebSocket connected'); socket.send('Hello Server'); }; socket.onmessage = function(event) { console.log('Message from server:', event.data); }; socket.onclose = function(event) { console.log('WebSocket disconnected'); }; socket.onerror = function(error) { console.error('WebSocket error:', error); }; // 2. 缺少重连机制 const socket = new WebSocket('ws://localhost:8080'); socket.onclose = function(event) { console.log('WebSocket disconnected'); // 没有重连逻辑 }; // 3. 缺少心跳机制 const socket = new WebSocket('ws://localhost:8080'); // 没有心跳逻辑,连接可能会被服务器或网络设备断开 // 4. 消息处理混乱 const socket = new WebSocket('ws://localhost:8080'); socket.onmessage = function(event) { const message = JSON.parse(event.data); if (message.type === 'chat') { console.log('Chat message:', message.data); } else if (message.type === 'notification') { console.log('Notification:', message.data); } else if (message.type === 'update') { console.log('Update:', message.data); } }; // 5. 缺少错误处理 const socket = new WebSocket('ws://localhost:8080'); socket.onerror = function(error) { console.error('WebSocket error:', error); // 没有错误处理逻辑 }; 

问题

  • 简单WebSocket连接,缺少重连机制
  • 缺少心跳机制,连接可能会被断开
  • 消息处理混乱,难以维护
  • 缺少错误处理,遇到错误时无法恢复
  • 没有状态管理,难以追踪连接状态

正确的做法

基本WebSocket实现

// 1. 基本WebSocket连接 class WebSocketClient { constructor(url) { this.url = url; this.socket = null; this.connected = false; this.reconnectAttempts = 0; this.maxReconnectAttempts = 5; this.reconnectDelay = 1000; this.messageHandlers = {}; this.heartbeatInterval = null; } connect() { this.socket = new WebSocket(this.url); this.socket.onopen = (event) => { console.log('WebSocket connected'); this.connected = true; this.reconnectAttempts = 0; this.startHeartbeat(); }; this.socket.onmessage = (event) => { this.handleMessage(event.data); }; this.socket.onclose = (event) => { console.log('WebSocket disconnected'); this.connected = false; this.stopHeartbeat(); this.reconnect(); }; this.socket.onerror = (error) => { console.error('WebSocket error:', error); }; } reconnect() { if (this.reconnectAttempts < this.maxReconnectAttempts) { this.reconnectAttempts++; console.log(`Attempting to reconnect... (${this.reconnectAttempts}/${this.maxReconnectAttempts})`); setTimeout(() => { this.connect(); }, this.reconnectDelay * this.reconnectAttempts); } else { console.error('Max reconnect attempts reached'); } } startHeartbeat() { this.heartbeatInterval = setInterval(() => { if (this.connected) { this.send({ type: 'heartbeat' }); } }, 30000); // 30秒发送一次心跳 } stopHeartbeat() { if (this.heartbeatInterval) { clearInterval(this.heartbeatInterval); this.heartbeatInterval = null; } } send(data) { if (this.connected) { this.socket.send(JSON.stringify(data)); } else { console.error('WebSocket not connected'); } } on(type, handler) { if (!this.messageHandlers[type]) { this.messageHandlers[type] = []; } this.messageHandlers[type].push(handler); } handleMessage(data) { try { const message = JSON.parse(data); const { type, payload } = message; if (this.messageHandlers[type]) { this.messageHandlers[type].forEach(handler => { handler(payload); }); } } catch (error) { console.error('Error parsing message:', error); } } disconnect() { this.stopHeartbeat(); if (this.socket) { this.socket.close(); } } } // 使用 const wsClient = new WebSocketClient('ws://localhost:8080'); wsClient.connect(); // 监听消息 wsClient.on('chat', (payload) => { console.log('Chat message:', payload); }); wsClient.on('notification', (payload) => { console.log('Notification:', payload); }); // 发送消息 wsClient.send({ type: 'chat', payload: { message: 'Hello', user: 'John' } }); 

React中使用WebSocket

import React, { useEffect, useCallback, useRef, useState } from 'react'; function WebSocketComponent() { const [messages, setMessages] = useState([]); const [input, setInput] = useState(''); const wsClientRef = useRef(null); // 初始化WebSocket连接 useEffect(() => { wsClientRef.current = new WebSocketClient('ws://localhost:8080'); wsClientRef.current.connect(); // 监听消息 wsClientRef.current.on('chat', (payload) => { setMessages(prev => [...prev, payload]); }); // 清理函数 return () => { if (wsClientRef.current) { wsClientRef.current.disconnect(); } }; }, []); // 发送消息 const handleSend = useCallback(() => { if (input.trim() && wsClientRef.current) { wsClientRef.current.send({ type: 'chat', payload: { message: input, user: 'Current User' } }); setInput(''); } }, [input]); return ( <div> <div className="messages"> {messages.map((message, index) => ( <div key={index} className="message"> <strong>{message.user}:</strong> {message.message} </div> ))} </div> <div className="input-area"> <input type="text" value={input} onChange={(e) => setInput(e.target.value)} onKeyPress={(e) => e.key === 'Enter' && handleSend()} /> <button onClick={handleSend}>Send</button> </div> </div> ); } export default WebSocketComponent; 

高级WebSocket实现

// 1. 带认证的WebSocket class AuthWebSocketClient extends WebSocketClient { constructor(url, token) { super(url); this.token = token; } connect() { this.socket = new WebSocket(`${this.url}?token=${this.token}`); // 其他逻辑与WebSocketClient相同 } } // 2. 带重试机制的WebSocket class RetryWebSocketClient extends WebSocketClient { constructor(url, options = {}) { super(url); this.maxReconnectAttempts = options.maxReconnectAttempts || 10; this.reconnectDelay = options.reconnectDelay || 1000; this.exponentialBackoff = options.exponentialBackoff || true; } reconnect() { if (this.reconnectAttempts < this.maxReconnectAttempts) { this.reconnectAttempts++; const delay = this.exponentialBackoff ? this.reconnectDelay * Math.pow(2, this.reconnectAttempts - 1) : this.reconnectDelay; console.log(`Attempting to reconnect... (${this.reconnectAttempts}/${this.maxReconnectAttempts})`); setTimeout(() => { this.connect(); }, delay); } else { console.error('Max reconnect attempts reached'); } } } // 3. 消息队列 class QueueWebSocketClient extends WebSocketClient { constructor(url) { super(url); this.messageQueue = []; } connect() { super.connect(); this.socket.onopen = (event) => { console.log('WebSocket connected'); this.connected = true; this.reconnectAttempts = 0; this.startHeartbeat(); // 发送队列中的消息 this.flushQueue(); }; } send(data) { if (this.connected) { this.socket.send(JSON.stringify(data)); } else { // 将消息加入队列 this.messageQueue.push(data); console.log('WebSocket not connected, message queued'); } } flushQueue() { if (this.connected && this.messageQueue.length > 0) { console.log(`Flushing ${this.messageQueue.length} queued messages`); this.messageQueue.forEach(message => { this.socket.send(JSON.stringify(message)); }); this.messageQueue = []; } } } 

最佳实践

// 1. WebSocket服务端实现(Node.js) const WebSocket = require('ws'); const server = new WebSocket.Server({ port: 8080 }); const clients = new Set(); server.on('connection', (socket) => { console.log('Client connected'); clients.add(socket); // 发送欢迎消息 socket.send(JSON.stringify({ type: 'system', payload: 'Welcome to the WebSocket server!' })); // 处理消息 socket.on('message', (message) => { try { const parsedMessage = JSON.parse(message); console.log('Received message:', parsedMessage); // 处理心跳 if (parsedMessage.type === 'heartbeat') { socket.send(JSON.stringify({ type: 'heartbeat' })); return; } // 广播消息 clients.forEach(client => { if (client.readyState === WebSocket.OPEN) { client.send(JSON.stringify(parsedMessage)); } }); } catch (error) { console.error('Error parsing message:', error); } }); // 处理断开连接 socket.on('close', () => { console.log('Client disconnected'); clients.delete(socket); }); // 处理错误 socket.on('error', (error) => { console.error('Socket error:', error); }); }); console.log('WebSocket server running on port 8080'); // 2. 前端WebSocket管理 class WebSocketManager { static instance = null; constructor() { this.clients = {}; } static getInstance() { if (!WebSocketManager.instance) { WebSocketManager.instance = new WebSocketManager(); } return WebSocketManager.instance; } createClient(name, url, options = {}) { const client = new QueueWebSocketClient(url, options); this.clients[name] = client; client.connect(); return client; } getClient(name) { return this.clients[name]; } removeClient(name) { if (this.clients[name]) { this.clients[name].disconnect(); delete this.clients[name]; } } disconnectAll() { Object.values(this.clients).forEach(client => { client.disconnect(); }); this.clients = {}; } } // 使用 const wsManager = WebSocketManager.getInstance(); const chatClient = wsManager.createClient('chat', 'ws://localhost:8080'); const notificationClient = wsManager.createClient('notification', 'ws://localhost:8080'); // 3. 错误处理和监控 class MonitoredWebSocketClient extends WebSocketClient { constructor(url) { super(url); this.errorCount = 0; this.maxErrorCount = 10; } handleError(error) { this.errorCount++; console.error('WebSocket error:', error); if (this.errorCount > this.maxErrorCount) { console.error('Max error count reached, disconnecting'); this.disconnect(); } } socket.onerror = (error) => { this.handleError(error); }; } 

毒舌点评

WebSocket确实很重要,但我见过太多开发者滥用这个特性,导致应用变得过于复杂。

想象一下,当你为了实现实时通信,使用了WebSocket,结果发现服务器负载过高,这真的值得吗?

还有那些过度使用WebSocket的开发者,为了实时性而使用WebSocket,结果发现轮询可能更加适合他们的场景。

所以,在使用WebSocket时,一定要根据实际需求来决定。不要为了使用WebSocket而使用,要选择最适合的方案。

当然,对于需要实时通信的应用来说,WebSocket是必要的。但对于不需要实时性的应用,轮询可能更加简单和可靠。

最后,记住一句话:WebSocket的目的是为了实现实时通信,而不是为了炫技。如果你的WebSocket实现导致应用变得更复杂或更不可靠,那你就失败了。

Read more

智能家居物联网平台

智能家居物联网平台

智能家居物联网平台 基于微服务+微信小程序实现的物联网平台,目前处于软件实现阶段:模拟了智能灯和温湿度感应器的使用状态,后续打算基于STM32制作硬件并接入,接入硬件后会将微信小程序改成Android版本 Gitee地址:仓库链接 技术栈 后端 * JDK:21 * Spring Boot:3.2.5 * Spring Cloud:2023.0.1 * Spring Cloud Alibaba:2023.0.1.0 * Spring Data JPA * Nacos:2.5.2 * MQTT * WebSocket 数据库 * MySQL:8.0 * Redis:6.2.x * Influxdb:2.8.

开源:AI+无人机巡检系统项目调研

主流开源AI无人机巡检项目调研 本部分系统梳理了当前主流的开源无人机巡检相关项目,涵盖飞控系统、地面站软件、AI视觉识别、数据处理等多个技术栈,为商业化产品开发提供技术选型参考。 一、飞控与地面站开源项目 1.1 PX4 Autopilot 项目地址:github.com/PX4/PX4-Autopilot 开源协议:BSD 3-Clause 项目简介:由Dronecode基金会(Linux基金会旗下)维护的专业级开源自动驾驶仪软件,是全球最广泛使用的无人机飞控系统之一。支持多旋翼、固定翼、垂直起降等多种机型,广泛应用于工业无人机和科研领域。 核心能力:飞行控制、任务规划、传感器融合、MAVLink通信协议、硬件抽象层、模块化架构 1.2 ArduPilot 项目地址:github.com/ArduPilot/ardupilot 开源协议:GPLv3 项目简介:历史最悠久的开源自动驾驶仪项目,社区活跃度极高。

一文说清FPGA如何实现高速数字信号处理

FPGA如何“硬刚”高速数字信号处理?从电路思维讲透设计本质 你有没有遇到过这样的场景: 一个实时频谱监测系统,要求每秒处理2.5亿个采样点,CPU跑得风扇狂转却依然延迟爆表; 或者在5G基站中,需要对上百路信号同时做滤波、变频和FFT——传统处理器根本扛不住这数据洪流。 这时候,工程师往往会说出那句经典台词:“这个任务,得用FPGA来搞。” 但问题是: 为什么是FPGA?它凭什么能“硬刚”这么猛的数字信号处理(DSP)任务? 今天我们就抛开那些教科书式的罗列与套话,从真实工程视角出发,把FPGA实现高速DSP这件事,掰开了揉碎了讲清楚。不堆术语,不画大饼,只说你能听懂、能上手、能优化的硬核逻辑。 一、别再拿CPU那一套想问题:FPGA的本质是“把算法变成电路” 我们先来问一个关键问题: 同样是执行 y = a * x + b 这个表达式,CPU 和 FPGA 到底有什么不同? * CPU :取指令

Flutter 三方库 shelf_modular 的鸿蒙化适配指南 - 掌控服务器路由资产、精密模块治理实战、鸿蒙级服务端专家

Flutter 三方库 shelf_modular 的鸿蒙化适配指南 - 掌控服务器路由资产、精密模块治理实战、鸿蒙级服务端专家

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 shelf_modular 的鸿蒙化适配指南 - 掌控服务器路由资产、精密模块治理实战、鸿蒙级服务端专家 在鸿蒙跨平台应用执行高级服务端管理与多维 Shelf 路由资产指控(如构建一个支持全场景秒级交互的鸿蒙大型全量后端服务中枢、处理海量 API Route Payloads 的语义认领或是实现一个具备极致指控能力的资产管理后台路由审计中心)时,如果仅仅依赖官方的基础 Shelf 处理器或者是极其繁琐的手动路由映射,极易在处理“由于模块嵌套导致的资产认领偏移”、“高频服务请求下的认领假死”或“由于多语言环境导致的符号解析冲突死结”时陷入研发代码服务端逻辑崩溃死循环。如果你追求的是一种完全对齐现代模块化标准、支持全量高度可定制路由(Modular-driven Backend)且具备极致指控确定性的方案。今天我们要深度解析的 shelf_modular——一个专注于解决“服务端资产标准化认领与模块化解耦”痛点的顶级工具库,正是帮你打造“鸿蒙超