NodeJs 实现 WebSocket 即时通讯(版本二)

服务端代码
websocket.js
'use strict'
const WebSocket = require('ws');
const connections = new Map();
const Constr = function(port) {
const self = this;
self.webSocket = new WebSocket.Server({ port: port });
};
Constr.prototype.connect = function() {
const self = this;
self.webSocket.on('connection', function connection(ws) {
try {
ws.on('message', function incoming(message) {
try {
if (connections.size > 2000) {
ws.send(1);
ws.close();
return;
}
ws.send(0);
if (connections.get(message) && connections.get(message).ws.readyState == WebSocket.OPEN) {
connections.get(message).date = Date.now();
connections.get(message).ws.isAlive = true;
console.log(message + ':上次心跳时间' + connections.get(message).date);
return;
}
connections.set(message, {
ws: ws,
date: Date.now(),
});
console.log('客户端imei:' + message + '握手、心跳成功!时间:' + connections.get(message).date);
ws.on('close', function close() {
console.log('连接关闭 ' + message)
// ws.reconnect();
if (connections.get(message)) {
connections.delete(message);
}
ws.close();
});
ws.on('error', function close() {
console.error(message + ':出现错误!强制下线');
connections.get(message)
.ws
.close();
});
// 服务器接受pong消息++huanglong,确定mtk接受ping,并pong服务器后打开
// ws.on('pong', function() {
// connections.get(message).date = Date.now();
// connections.get(message).ws.isAlive = true;
// console.log('上次接收pong时间' + connections.get(message).date);
// });
} catch (e) {
console.error('on message error:', e);
}
});
}catch (e) {
console.error('on connection error:', e);
}
});
self.webSocket.on('error',function(){
console.log('error');
});
};
Constr.prototype.send = function(obj) {
console.log('下推消息:' + JSON.stringify(obj));
const imeiArray = obj.imei.split(',');
for (let i = 0; i < imeiArray.length; i++) {
try {
if (connections.get(imeiArray[i]) && connections.get(imeiArray[i]).ws.readyState === WebSocket.OPEN) {
console.log(imeiArray[i] + ':连接状态' + connections.get(imeiArray[i]).ws.readyState);
connections.get(imeiArray[i]).ws.send(obj.str);
} else {
console.log('imei:' + imeiArray[i] + 'socket关闭!');
connections.delete(imeiArray[i]);
}
} catch (e) {
console.log('发送消息发生严重错误!imei:' + imeiArray[i]);
}
}
};
Constr.prototype.heartbeatCheck = function() {
console.log('心跳检查:当前握手连接数为' + connections.size + '客户端:' + connections.keys().toString());
if (connections.size === 0) {
return;
}
connections.forEach(function (value, key) {
if (Date.now() - value.date > 60000) {
connections.delete(key);
try {
value.ws.close();
}catch (e) {
console.error('close error', e);
}
}
// ++huanglong,暂时关闭ping机制,确定mtk接受ping,并pong服务器后打开
// if (value.ws.isAlive === false) return value.ws.terminate();
// value.ws.isAlive = false;
// value.ws.ping(function() {
// value.ws.send(2);
// });
});
};
// Constr.prototype.testyuyin2 = function() {
// connections.forEach(function(value) {
// value.ws.send('测试语音');
// });
// };
module.exports = Constr;
app.js
const Ws = require('./app/middleware/websocket');
const ws = new Ws(8080);
try {
ws.connect();
} catch (e) {
console.error('ws connect error:', e);
}
console.log("WebSocket建立完毕")
客户端
const WebSocket = require('ws');
var lockReconnect = false;//避免重复连接
var wsUrl = "ws://127.0.0.1:8080";
var ws;
var tt;
function createWebSocket() {
try {
ws = new WebSocket(wsUrl);
init();
} catch (e) {
console.log('catch' + e);
reconnect(wsUrl);
}
}
function init() {
ws.onclose = function () {
console.log('链接关闭');
reconnect(wsUrl);
};
ws.onerror = function () {
console.log('发生异常了');
reconnect(wsUrl);
};
ws.onopen = function () {
//心跳检测重置
heartCheck.start();
};
ws.onmessage = function (event) {
//拿到任何消息都说明当前连接是正常的
console.log('接收到消息' + JSON.stringify(event.data));
heartCheck.start();
}
}
function reconnect(url) {
if (lockReconnect) {
return;
};
lockReconnect = true;
//没连接上会一直重连,设置延迟避免请求过多
tt && clearTimeout(tt);
tt = setTimeout(function () {
createWebSocket(url);
lockReconnect = false;
}, 4000);
}
//心跳检测
var heartCheck = {
timeout: 3000,
timeoutObj: null,
serverTimeoutObj: null,
start: function () {
// console.log('start');
var self = this;
this.timeoutObj && clearTimeout(this.timeoutObj);
this.serverTimeoutObj && clearTimeout(this.serverTimeoutObj);
this.timeoutObj = setTimeout(function () {
//这里发送一个心跳,后端收到后,返回一个心跳消息,
ws.send("666666");
self.serverTimeoutObj = setTimeout(function () {
// console.log(111);
// console.log(ws);
ws.close();
// createWebSocket();
}, self.timeout);
}, this.timeout)
}
}
createWebSocket(wsUrl);
NodeJs 实现 WebSocket 即时通讯(版本二)的更多相关文章
- NodeJs 实现 WebSocket 即时通讯(版本一)
服务端代码 var ws = require("nodejs-websocket"); console.log("开始建立连接...") var game1 = ...
- HTML5+NodeJs实现WebSocket即时通讯
声明:本文为原创文章,如需转载,请注明来源WAxes,谢谢! 最近都在学习HTML5,做canvas游戏之类的,发现HTML5中除了canvas这个强大的工具外,还有WebSocket也很值得注意.可 ...
- 使用tomcat方式实现websocket即时通讯服务端讲解
使用tomcat方式实现websocket即时通讯服务端讲解 第一种方案:使用Tomcat的方式实现 tomcat版本要求:tomcat7.0+.需要支持Javaee7 导入javeee-api的ja ...
- Springboot 项目源码 Activiti6 工作流 vue.js html 跨域 前后分离 websocket即时通讯
特别注意: Springboot 工作流 前后分离 + 跨域 版本 (权限控制到菜单和按钮) 后台框架:springboot2.1.2+ activiti6.0.0+ mybaits+maven+接 ...
- java SSM框架 代码生成器 快速开发平台 websocket即时通讯 shiro redis
A代码编辑器,在线模版编辑,仿开发工具编辑器,pdf在线预览,文件转换编码 B 集成代码生成器 [正反双向](单表.主表.明细表.树形表,快速开发利器)+快速表单构建器 freemaker模版技术 , ...
- IdentityServer4 + SignalR Core +RabbitMQ 构建web即时通讯(二)
IdentityServer4 + SignalR Core +RabbitMQ 构建web即时通讯(二) IdentityServer4 用户中心生成数据库 上文已经创建了所有的数据库上下文迁移代码 ...
- java SSM 框架 代码生成器 websocket即时通讯 shiro redis
1. 权限管理:点开二级菜单进入三级菜单显示 角色(基础权限)和按钮权限 角色(基础权限): 分角色组和角色,独立分配菜单权限和增删改查权限. 按钮权限: 给角色分配按钮权限. ...
- [开源] .NETCore websocket 即时通讯组件---ImCore
前言 ImCore 是一款 .NETCore 下利用 WebSocket 实现的简易.高性能.集群即时通讯组件,支持点对点通讯.群聊通讯.上线下线事件消息等众多实用性功能. 开源地址:https:// ...
- [重磅开源] 比SingleR更适合的websocket 即时通讯组件---ImCore开源了
有感而发 为什么说 SignalR 不合适做 IM? IM 的特点必定是长连接,轮训的功能用不上. 因为它是双工通讯的设计,用hub.invoke发送命令给服务端处理业务,其他就和 ajax 差不多, ...
随机推荐
- vue平行组件传值
平行组件传值 通过平行组件传值可以实现任何情境下的传值,包括(父传子,子传父) 代码示例 <!DOCTYPE html> <html lang="en"> ...
- php常用函数(第一版)
1.array_slice 作用:数组分页函数 案例:$output = array_slice ( $input , - 2 , 1 ); 2.array_column 作用:数组根据值取出一 ...
- linux shell 统计当前目录下的文件个数
shell 统计当前目录下文件个数,使用管道组合命令: ls -1 | wc -l 解释: ls -1 表示一行一个列出文件名. wc -l 表示打印统计的行数. 两个命令通过管道连在一起表示打印列出 ...
- Android_布局
<该文章参考各大博客以及书籍总结而来,如有问题欢迎指出^ ^> 一.五大传统布局+新布局 线性布局——LinearLayout 相对布局——RelativeLayout 帧布局——Fram ...
- 推荐一款可以丰富博文GIF免费录制工具——GifCam
在网上写博文,看别人添加gif图片很好奇,于是尝试制作并传递了一次,确实挺有意思的,于是分享一下. gifcam是一个绿色的,可以录制GIF动图的小软件,十分适合用来录制电脑的各种操作. 在这里附上工 ...
- 关于5G目前形式的浅谈
目前市面上已经有十几款5G手机已经上市了,但是到2019年底的覆盖率还是非常的低,所以很多换机的用户可以再等一等,2020应该是出来的都是5G手机,也都是支持双模的NA/NSA的两种组网模式,因为工信 ...
- VUE--插值的操作
一.vue常见的指令 v-once:保留第一次渲染结果 v-html :把html代码解析,只显示内容 v-pre :原样输出 v-cloak :解决文本闪烁问题 v-text :显示文本 二.v-b ...
- Java之微信公众号开发
这次以文本回复作为案例来讲解Java相关得微信公众号开发. 首先必须要有一个个人微信公众号 个人微信公众号相关的接口权限有限,不过用于个人学习体验一下足够了,如图: 然后进入微信公众后台,点击基本配置 ...
- 机器学习:weka中添加自己的分类和聚类算法
不管是实验室研究机器学习算法或是公司研发,都有需要自己改进算法的时候,下面就说说怎么在weka里增加改进的机器学习算法. 一 添加分类算法的流程 1 编写的分类器必须继承 Classifier或是Cl ...
- C# 控件 RichTextBox 显示行号,并且与Panel相互联动
我们在使用到WINFORM窗体工作中,要求RichTextBox 加入行号: 之前有看到大牛们写的,但是太复杂繁多,而且有用双TextBox进行联动,非常不错,今天我们尝试RichTextBox +P ...