一般网站建设公司株洲搜索引擎优化
web/
2025/10/4 23:19:58/
文章来源:
一般网站建设公司,株洲搜索引擎优化,在电商网站上做推广的技巧,做pc端网站渠道1 认识webSocket
WebSocket_ohana#xff01;的博客-CSDN博客
一#xff0c;什么是websocket
WebSocket是HTML5下一种新的协议#xff08;websocket协议本质上是一个基于tcp的协议#xff09;它实现了浏览器与服务器全双工通信#xff0c;能更好的节省服务器资源和带宽…1 认识webSocket
WebSocket_ohana的博客-CSDN博客
一什么是websocket
WebSocket是HTML5下一种新的协议websocket协议本质上是一个基于tcp的协议它实现了浏览器与服务器全双工通信能更好的节省服务器资源和带宽并达到实时通讯的目的Websocket是一个持久化的协议
WebSocket使得客户端和服务器之间的数据交换变得更加简单允许服务端主动向客户端推送数据。在WebSocket API中浏览器和服务器只需要完成一次握手两者之间就直接可以创建持久性的连接并进行双向数据传输。 2 技术选型 为什么选择webSocket WebSocket有以下特点 是真正的全双工方式建立连接后客户端与服务器端是完全平等的可以互相主动请求。而HTTP长连接基于HTTP是传统的客户端对服务器发起请求的模式。 HTTP长连接中每次数据交换除了真正的数据部分外服务器和客户端还要大量交换HTTP header信息交换效率很低。Websocket协议通过第一个request建立了TCP连接之后之后交换的数据都不需要发送 HTTP header就能交换数据这显然和原有的HTTP协议有区别所以它需要对服务器和客户端都进行升级才能实现主流浏览器都已支持HTML5 3 WebSocket Demo
前端服务
npm install websocket // websocket: ^1.0.32,新建 websocket.js 文件
node 安装
npm install ws // ws: ^7.3.0, vue项目中使用WebSocket_vue websocket服务端_weixin_43964779的博客-CSDN博客 开发者API
WebSocket() - Web API 接口参考 | MDN
client: scriptvar app new Vue({el: #app,data: {message: ,lists: [],ws: {},name: ,isShow: true,num: 0,roomid: ,uid: ,handle: {}},mounted() {},methods: {init() {this.ws new WebSocket(ws://127.0.0.1:3000)this.ws.onopen this.onOpenthis.ws.onmessage this.onMessagethis.ws.onclose this.onClosethis.ws.onerror this.onError},enter() {if (this.name.trim() ) {alert(用户名不得为空)return}this.init()this.isShow false},onOpen: function () {// console.log(open: this.ws.readyState);//ws.send(Hello fro,m client!)// 发起鉴权请求//this.ws.send(JSON.stringify({// event: auth,// message: //eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIx//MjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNT//E2MjM5MDIyfQ.//XbPfbIHMI6arZ3Y922BhjWgQzWXcXNrz0ogtVhfEd2o//}))this.ws.send(JSON.stringify({event: enter,message: this.name,roomid: this.roomid,uid: this.uid}))},onMessage: function (event) {// 当用户未进入聊天室则不接收消息if (this.isShow) {return}// 接收服务端发送过来的消息var obj JSON.parse(event.data)switch (obj.event) {case noauth:// 鉴权失败// 路由跳转到 /login 重新获取tokenbreak;case enter:// 当有一个新的用户进入聊天室this.lists.push(欢迎 obj.message 加入聊天室)break;case out:this.lists.push(obj.name 已经退出了聊天室)break;case heartbeat://this.checkServer() // timeInterval t// 可以注释掉以下心跳状态主动测试服务端是否会断开客户端的连接this.ws.send(JSON.stringify({event: heartbeat,message: pong}))breakdefault:if (obj.name ! this.name) {// 接收正常的聊天this.lists.push(obj.name : obj.message)}}this.num obj.num},onClose: function () {// 当链接主动断开的时候触发close事件console.log(close: this.ws.readyState);console.log(已关闭websocket);this.ws.close()},onError: function () {// 当连接失败时触发error事件console.log(error: this.ws.readyState);console.log(websocket连接失败);// 连接失败之后1s进行断线重连var _this thissetTimeout(function () {_this.init()}, 1000)},// 发送消息send: function () {this.lists.push(this.name : this.message)this.ws.send(JSON.stringify({event: message,message: this.message,name: this.name}))this.message },checkServer: function () {var _this thisclearTimeout(this.handle)this.handle setTimeout(function () {_this.onClose()setTimeout(function () {_this.init()}, 1000)// 设置1ms的时延调试在服务器测未及时响应时客户端的反应}, 30000 1000)}}})/script
Server:
package.json:
{name: server,version: 1.0.0,description: ,main: index.js,scripts: {start: nodemon src/index.js},keywords: [],author: ,license: ISC,dependencies: {bluebird: ^3.7.2,jsonwebtoken: ^8.5.1,redis: ^2.8.0,ws: ^7.2.1},devDependencies: {nodemon: ^2.0.2}
}index.js:
const WebSocket require(ws)
const wss new WebSocket.Server({ port: 3000 })
// const jwt require(jsonwebtoken)
const {getValue, setValue, existKey} require(./config/RedisConfig)const timeInterval 30000
// 多聊天室的功能
// roomid - 对应相同的roomid进行广播消息
let group {}// const run async () {
// setValue(imooc, hello)
// const result await getValue(imooc)
// console.log(TCL: run - result, result)
// }// run()
const prefix imooc-
wss.on(connection, function connection (ws) {// 初始的心跳连接状态ws.isAlive trueconsole.log(one client is connected);// 接收客户端的消息ws.on(message, async function(msg) {const msgObj JSON.parse(msg)const roomid prefix (msgObj.roomid ? msgObj.roomid : ws.roomid)if (msgObj.event enter) {// 当用户进入之后需要判断用户的房间是否存在// 如果用户的房间不存在则在redis中创建房间号用户保存用户信息// 主要是用于统计房间里的人数用于后面进行消息发送ws.name msgObj.messagews.roomid msgObj.roomidws.uid msgObj.uidconsole.log(TCL: connection - ws.uid, ws.uid)// 判断redis中是否有对应的roomid的键值const result await existKey(roomid)if (result 0) {// 初始化一个房间数据setValue(roomid, ws.uid)} else {// 已经存在该房间缓存数据const arrStr await getValue(roomid)let arr arrStr.split(,)if (arr.indexOf(ws.uid) -1) {setValue(roomid, arrStr , ws.uid)}}if (typeof group[ws.roomid] undefined) {group[ws.roomid] 1} else {group[ws.roomid] }}// // 鉴权// if (msgObj.event auth) {// jwt.verify(msgObj.message, secret, (err, decode) {// if (err) {// // websocket返回前台鉴权失败消息// ws.send(JSON.stringify({// event: noauth,// message: please auth again// })) // console.log(auth error);// return// } else {// // 鉴权通过// console.log(decode);// ws.isAuth true// return // }// })// return// }// // 拦截非鉴权的请求// if (!ws.isAuth) {// return// }// 心跳检测if (msgObj.event heartbeat msgObj.message pong) {ws.isAlive truereturn}// 广播消息// 获取房间里所有的用户信息const arrStr await getValue(roomid)let users arrStr.split(,)wss.clients.forEach(async (client) {// 判断非自己的客户端if (client.readyState WebSocket.OPEN client.roomid ws.roomid) {msgObj.name ws.namemsgObj.num group[ws.roomid]client.send(JSON.stringify(msgObj))// 排队已经发送了消息了客户端 - 在线if (users.indexOf(client.uid) ! -1) {users.splice(users.indexOf(client.uid), 1)}// 消息缓存信息取redis中的uid数据let result await existKey(ws.uid)if (result ! 0) {// 存在未发送的离线消息数据let tmpArr await getValue(ws.uid)let tmpObj JSON.parse(tmpArr)let uid ws.uidif (tmpObj.length 0) {let i []// 遍历该用户的离线缓存数据// 判断用户的房间id是否与当前一致tmpObj.forEach((item) {if (item.roomid client.roomid uid client.uid) {client.send(JSON.stringify(item))i.push(item)}})// 删除已经发送的缓存消息数据if (i.length 0) {i.forEach((item) {tmpObj.splice(item, 1)})}setValue(ws.uid, JSON.stringify(tmpObj))}}}})// 断开了与服务端连接的用户的id并且其他的客户端发送了消息if (users.length 0 msgObj.event message) {users.forEach(async function(item) {const result await existKey(item)if (result ! 0) {// 说明已经存在其他房间该用户的离线消息数据let userData await getValue(item)let msgs JSON.parse(userData)msgs.push({roomid: ws.roomid,...msgObj})setValue(item, JSON.stringify(msgs))} else {// 说明先前这个用户一直在线并且无离线消息数据setValue(item, JSON.stringify([{roomid: ws.roomid,...msgObj}]))}})}})// 当ws客户端断开链接的时候ws.on(close, function() {if (ws.name) {group[ws.roomid] --}let msgObj {}// 广播消息wss.clients.forEach((client) {// 判断非自己的客户端if (client.readyState WebSocket.OPEN ws.roomid client.roomid) {msgObj.name ws.namemsgObj.num group[ws.roomid]msgObj.event outclient.send(JSON.stringify(msgObj))}})})
})// setInterval(() {
// wss.clients.forEach((ws) {
// if (!ws.isAlive ws.roomid) {
// group[ws.roomid] --
// delete ws[roomid]
// return ws.terminate()
// }
// // 主动发送心跳检测请求
// // 当客户端返回了消息之后主动设置flag为在线
// ws.isAlive false
// ws.send(JSON.stringify({
// event: heartbeat,
// message: ping,
// num: group[ws.roomid]
// }))
// })
// }, timeInterval) 4 心跳检测断线重连
服务器先发-客户端-服务器 ∞
心跳检测
服务ping-客户端 服务器端有定时器 如果没有收到 会在下次遍历中关闭与该客户的服务 ws.terminate() //终止发送 退出了本次连接
客户段Clinet
客户段在定时器中加入 断开重连的代码在下次服务器发送过来的PIng 的代码中 清除掉上次的定时器同时就清除了上次心跳检查的断开代码然后发送pong-服务器服务器收到后继续大发送ping-客户端当本次请求一直未收到ping时 心跳检查的定时器没有被清除 就会执行close方法关闭本次连接并重新Init新的链接这就是断线重连。 服务器端 定时器
setInterval(() { //定时器wss.clients.forEach((ws) {if (!ws.isAlive ws.roomid) { //客户段终止group[ws.roomid] -- delete ws[roomid]return ws.terminate() //终止发送 退出了本次连接}// 主动发送心跳检测请求// 当客户端返回了消息之后主动设置flag为在线ws.isAlive falsews.send(JSON.stringify({event: heartbeat,message: ping,num: group[ws.roomid]}))})
}, timeInterval)
客户端 心跳检查与 断线重连
//心跳检查case heartbeat://this.checkServer() // timeInterval t 如果一直接收到ping 那么这次的请求就会删除上次的定时器 定时器不会被执行// 可以注释掉以下心跳状态主动测试服务端是否会断开客户端的连接this.ws.send(JSON.stringify({event: heartbeat,message: pong}))break//检查心跳 checkServer: function () {var _this thisclearTimeout(this.handle) //清除计时器this.handle setTimeout(function () {_this.onClose()setTimeout(function () {_this.init()}, 1000)// 设置1ms的时延调试在服务器测未及时响应时客户端的反应}, 30000 1000)} springboot整合webSocket看完即入门_springboot websocket_hmb↑的博客-CSDN博客 webSocket前端开发实现心跳检测机制_前端心跳检测_mayuan2011的博客-CSDN博客 为什么要使用webSocket以及心跳检测机制
在使用webSocket的过程中如果遇到网络断开服务端并没有触发onclose事件就会出现此状况服务端会继续向客户端发送多余的连接并且这些数据会丢失。 因此就需要一种机制来检测客户端和服务端是否处于正常的连接状态因此就有了webSocket的心跳检测机制即如果有心跳则说客户端和服务端的连接还存在无心跳相应则说明链接已经断开需要采取重新连接等措施。 5 总结
WebSocket是一种在单个TCP连接上进行全双工通信的协议。WebSocket使得客户端和服务器之间的数据交换变得更加简单允许服务端主动向客户端推送数据。在WebSocket API中浏览器和服务器只需要完成一次握手两者之间就直接可以创建持久性的连接并进行双向数据传输。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/87029.shtml
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!