前端面试练习24.3.6

2024-03-07 18:04

本文主要是介绍前端面试练习24.3.6,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言:

今天就是复习回顾一下websocket,之前有个AI项目使用到了这个,有点久远了,回顾一下大致过程和难点。

一些知识点:

1.单工,半双工,全双工

单工:

数据只能沿着一个方向传递,例如电视广播等。

优点:实现简单

缺点:传输效率低

半双工:

数据可以在双方之间进行传递,但是不能同时进行,必须有个 发送/接收 角色的转换,一方发送完成另一方才能发送。比如对讲机。

全双工:

数据可以同时在两个方向上传输,通信的两个设备之间可以同时的收发消息,无需等待对方完成,极大地提高了数据的传输效率,典型用例就是电话。

2.websocket可能遇到的难点

参考WebSocket项目中难点与解决方法_websocket客服聊天 难点-CSDN博客

难点一:连接的建立与保持

1.采用连接池

WebSocket初始连接负担较大,主要体现在频繁的连接建立和保持连接的开销较高。

采用连接池,服务端解决

引入了websocket-pool库,通过维护连接池,成功实现了连接的复用。这极大地降低了频繁建立和关闭连接的开销,提升了性能。

 

2.心跳机制引入

 实施了定时的心跳机制,周期性地向服务器发送心跳消息,确保连接保持活跃。这有效防止了连接被自动关闭,提高了连接的可靠性。

3.长连接,适当延长连接时常

比如在连接时,设置超时时长5秒,5秒内没有得到服务器回应,则出现超时提示​​​​​​​

  

难点二:错误处理与断线重连

1.错误处理

 监听onerror事件,当 WebSocket 发生错误时,执行回调函数捕获错误信息​​​​​​​

2.短线重连

 自动断线重连机制,采用了指数退避算法。通过逐渐增加重连的间隔时间,我们成功避免了频繁尝试重新建立连接,确保了连接的稳定性和用户体验。

难点三:性能的一些优化

1.消息压缩

目前还没解决

2.并发限制

连接池设置最大数量

3.消息队列

目前没解决

websocket实战:

服务端创建websocket服务

创建node项目,安装相关依赖

npm init -y
npm install express
npm install ws
const WebSocket = require('ws');
const PORT = 8080;
const URL = 'ws://127.0.0.1:8080';
const wss = new WebSocket.Server({ port: PORT });
const clients = new Map();wss.on('connection', function connection(ws) {console.log('Client connected');ws.on('message', function incoming(message) {const data = JSON.parse(message);if (typeof data !== 'string') {let targetUser = data.toUser// 如果客户端之前没有存储在 clients Map 中,则存储客户端连接if (!clients.has(data.user)) {clients.set(data.user, ws);}console.log(clients.keys())// 将格式化后的消息发送给客户端const formattedMessage = `${data.user}:  - ${data.time}<br>${data.message}`;if (clients.has(targetUser)) {ws.send(formattedMessage);const clientSocket = clients.get(targetUser);// 检查连接是否已经打开if (clientSocket.readyState === WebSocket.OPEN) {// 发送消息给目标用户clientSocket.send(formattedMessage);} else {clientSocket.send(`连接到用户 ${targetUser} 的WebSocket已关闭。`);}} else {ws.send(`用户 ${targetUser} 不在线或未连接到WebSocket服务器。`);}console.log('Received:', formattedMessage);}});// 当完成后释放连接回连接池// webSocketPool.releaseConnection(connection);
});console.log('WebSocket server is running on port 8080');
console.log(`WebSocket 服务器正在监听 127.0.0.1:${PORT}`);

客户端编写连接代码

<!DOCTYPE html>
<html><head><meta charset='utf-8'><title>前端学习</title><style>* {margin: 0px;padding: 0px;}</style>
</head><body>写一个聊天显示的窗口,以及信息发送输入框和发送按钮<div id='chat' style="display: flex;justify-content: space-around;"><div class="box1"><div id='user1_messages' style="width: 600px;height: 300px;border:1px solid black"><h1>WebSocket1 user:<span id="user1_h1" style="color: rgb(232, 154, 66);"></span></h1></div><input type='text' id='messageInput1' autocomplete='off' placeholder='Type your message here...'><button id='sendButton1'>Send</button></div><div class="box2"><div id='user2_messages' style="width: 600px;height: 300px;border:1px solid black"><h1>WebSocket2 user:<span id="user2_h1" style="color: rgb(232, 154, 66);"></span></h1></div><input type='text' id='messageInput2' autocomplete='off' placeholder='Type your message here...'><button id='sendButton2'>Send</button></div></div><script type='text/javascript'>let data = [{user: 'xhc',message: '',time: '',toUser: ''}, {user: 'slj',message: '',time: '',toUser: ''},{user: 'aaa',message: '',time: '',toUser: ''}]let mess1 = document.getElementById('user1_messages');let mess2 = document.getElementById('user2_messages');let user1_h1 = document.getElementById('user1_h1');user1_h1.innerHTML = `${data[0].user}`let input1 = document.getElementById('messageInput1');let input2 = document.getElementById('messageInput2');let user2_h1 = document.getElementById('user2_h1');user2_h1.innerHTML = `${data[1].user}`let send1 = document.getElementById('sendButton1');let send2 = document.getElementById('sendButton2');const URL = 'ws://localhost:8080'const connect1 = () => {// 创建 WebSocket 连接const socket = new WebSocket(URL);let timer; // Variable to store the interval// 当连接打开时执行的操作socket.onopen = function (event) {console.log('WebSocket1 连接已建立');data[0].message = input1.value;data[0].time = new Date().toLocaleString();data[0].toUser = 'slj';// 发送消息到服务器socket.send(JSON.stringify(data[0]));input1.value = '';// 设置定时器,每隔30秒执行一次指定的回调函数timer = setInterval(() => {// 检查 WebSocket 连接的当前状态是否为 OPENif (socket.readyState === WebSocket.OPEN) {// 如果连接处于 OPEN 状态,则通过连接对象发送心跳消息 'keep-alive'socket.send(JSON.parse('keep-alive'));}}, 30000); // 每 30 秒发送一次心跳};// 当接收到消息时执行的操作socket.onmessage = function (event) {let reply = event.data;mess1.innerHTML = `${mess1.innerHTML} <br>${reply}`;console.log('来自服务器的消息1:', reply);};// 当连接关闭时执行的操作socket.onclose = function (event) {mess1.innerHTML = 'WebSocket1 连接已关闭';console.log('WebSocket1 连接已关闭');// 清除定时器clearInterval(timer);};};send1.addEventListener('click', connect1)let reconnectDelay = 2000; // 初始重连延迟为 2 秒let reconnectCount = 0; // 当前重连次数const maxReconnectCount = 7; // 最大重连次数为 7const maxReconnectDelay = 60000; // 最大重连延迟为 60 秒const connect2 = () => {let socket; // WebSocket 连接对象let timer; // 用于心跳的定时器let timeoutTimer; // 连接超时的定时器// 定义超时操作函数function handleTimeout() {console.error('WebSocket2 连接超时');// 关闭连接if (socket) {socket.close();}}// 定义重新连接函数function reconnect() {reconnectCount++;console.log(`尝试重新连接 WebSocket2,第 ${reconnectCount} 次...`);clearTimeout(timeoutTimer); // 清除超时计时器if (reconnectCount <= maxReconnectCount) {setTimeout(connect2, reconnectDelay); // 重新连接// 更新重连延迟时间,采用指数退避算法reconnectDelay = Math.min(2 * reconnectDelay, maxReconnectDelay);} else {console.error(`WebSocket2 重连失败,已达到最大重连次数 (${maxReconnectCount} 次)`);}}// 创建 WebSocket 连接socket = new WebSocket(URL);// 当连接打开时执行的操作socket.onopen = function (event) {console.log('WebSocket2 连接已建立');data[1].message = input2.value;data[1].time = new Date().toLocaleString();data[1].toUser = 'xhc';// 发送消息到服务器if (socket.readyState === WebSocket.OPEN) {// 如果连接处于 OPEN 状态,则通过连接对象发送消息socket.send(JSON.stringify(data[1]));} else {console.log('WebSocket2 连接未建立/连接建立失败/连接关闭');}input2.value = '';// 设置定时器,每隔30秒执行一次指定的回调函数timer = setInterval(() => {// 检查 WebSocket 连接的当前状态是否为 OPENif (socket.readyState === WebSocket.OPEN) {// 如果连接处于 OPEN 状态,则通过连接对象发送心跳消息 'keep-alive'socket.send(JSON.parse('keep-alive'));}}, 30000); // 每 30 秒发送一次心跳// 设置超时计时器timeoutTimer = setTimeout(handleTimeout, reconnectDelay);};// 当接收到消息时执行的操作socket.onmessage = function (event) {mess2.innerHTML = `${mess2.innerHTML} <br>${event.data}`;console.log('来自服务器的消息2:', event.data);// 清除超时计时器clearTimeout(timeoutTimer);};// 当连接关闭时执行的操作socket.onclose = function (event) {mess2.innerHTML = 'WebSocket2 连接已关闭';console.log('WebSocket2 连接已关闭');// 清除定时器if (timer) {clearInterval(timer);}};// 当发生错误时执行的操作socket.onerror = function (error) {console.error('WebSocket2 连接发生错误:', error);// 清除超时计时器clearTimeout(timeoutTimer);clearInterval(timer);};};send2.addEventListener('click', connect2)</script>
</body></html>

这篇关于前端面试练习24.3.6的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/784359

相关文章

SpringBoot返回文件让前端下载的几种方式

《SpringBoot返回文件让前端下载的几种方式》文章介绍了开发中文件下载的两种常见解决方案,并详细描述了通过后端进行下载的原理和步骤,包括一次性读取到内存和分块写入响应输出流两种方法,此外,还提供... 目录01 背景02 一次性读取到内存,通过响应输出流输出到前端02 将文件流通过循环写入到响应输出流

SpringBoot+Vue3整合SSE实现实时消息推送功能

《SpringBoot+Vue3整合SSE实现实时消息推送功能》在日常开发中,我们经常需要实现实时消息推送的功能,这篇文章将基于SpringBoot和Vue3来简单实现一个入门级的例子,下面小编就和大... 目录前言先大概介绍下SSE后端实现(SpringBoot)前端实现(vue3)1. 数据类型定义2.

前端Visual Studio Code安装配置教程之下载、汉化、常用组件及基本操作

《前端VisualStudioCode安装配置教程之下载、汉化、常用组件及基本操作》VisualStudioCode是微软推出的一个强大的代码编辑器,功能强大,操作简单便捷,还有着良好的用户界面,... 目录一、Visual Studio Code下载二、汉化三、常用组件1、Auto Rename Tag2

vite搭建vue3项目的搭建步骤

《vite搭建vue3项目的搭建步骤》本文主要介绍了vite搭建vue3项目的搭建步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录1.确保Nodejs环境2.使用vite-cli工具3.进入项目安装依赖1.确保Nodejs环境

Nginx搭建前端本地预览环境的完整步骤教学

《Nginx搭建前端本地预览环境的完整步骤教学》这篇文章主要为大家详细介绍了Nginx搭建前端本地预览环境的完整步骤教学,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录项目目录结构核心配置文件:nginx.conf脚本化操作:nginx.shnpm 脚本集成总结:对前端的意义很多

前端缓存策略的自解方案全解析

《前端缓存策略的自解方案全解析》缓存从来都是前端的一个痛点,很多前端搞不清楚缓存到底是何物,:本文主要介绍前端缓存的自解方案,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、为什么“清缓存”成了技术圈的梗二、先给缓存“把个脉”:浏览器到底缓存了谁?三、设计思路:把“发版”做成“自愈”四、代码

通过React实现页面的无限滚动效果

《通过React实现页面的无限滚动效果》今天我们来聊聊无限滚动这个现代Web开发中不可或缺的技术,无论你是刷微博、逛知乎还是看脚本,无限滚动都已经渗透到我们日常的浏览体验中,那么,如何优雅地实现它呢?... 目录1. 早期的解决方案2. 交叉观察者:IntersectionObserver2.1 Inter

Vue3视频播放组件 vue3-video-play使用方式

《Vue3视频播放组件vue3-video-play使用方式》vue3-video-play是Vue3的视频播放组件,基于原生video标签开发,支持MP4和HLS流,提供全局/局部引入方式,可监听... 目录一、安装二、全局引入三、局部引入四、基本使用五、事件监听六、播放 HLS 流七、更多功能总结在 v

JS纯前端实现浏览器语音播报、朗读功能的完整代码

《JS纯前端实现浏览器语音播报、朗读功能的完整代码》在现代互联网的发展中,语音技术正逐渐成为改变用户体验的重要一环,下面:本文主要介绍JS纯前端实现浏览器语音播报、朗读功能的相关资料,文中通过代码... 目录一、朗读单条文本:① 语音自选参数,按钮控制语音:② 效果图:二、朗读多条文本:① 语音有默认值:②

vue监听属性watch的用法及使用场景详解

《vue监听属性watch的用法及使用场景详解》watch是vue中常用的监听器,它主要用于侦听数据的变化,在数据发生变化的时候执行一些操作,:本文主要介绍vue监听属性watch的用法及使用场景... 目录1. 监听属性 watch2. 常规用法3. 监听对象和route变化4. 使用场景附Watch 的