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 差不多, ...
随机推荐
- spring后台重定向方式
1.直接返回值中加重定向:"redirect:要访问的网址"; public String updateOrAddProject() { return "redirect ...
- Java8新特性——lambda函数式编程
一.遍历循环 /** * @author jiaqing.xu@hand-china.com * @version 1.0 * @name * @description 循环遍历 * @date 20 ...
- 痞子衡嵌入式:飞思卡尔i.MX RTyyyy系列MCU特性那些事(5)- 划时代新品RT1170
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是飞思卡尔i.MX RTyyyy系列MCU的划时代新品i.MXRT1170. 自2017年开始,每年的6月25日恩智浦都会在北京举行微控制 ...
- Stanford公开课《编译原理》学习笔记(2)递归下降法
目录 一. Parse阶段 CFG Recursive Descent(递归下降遍历) 二. 递归下降遍历 2.1 预备知识 2.2 多行语句的处理思路 2.3 简易的文法定义 2.4 文法产生式的代 ...
- python selenium模拟登陆qq空间
不多说.直接上代码 from selenium import webdriver driver = webdriver.Chrome() driver.get('http://qzone.qq.com ...
- 什么是回流(重排 reflow)?什么是重绘(repaint)?如何减少回流、重绘?
什么是回流(重排 reflow)? 回流(重排 reflow):对DOM树进行渲染,只要修改DOM或修改元素的形状大小,就会触发reflow,reflow的时候,浏览器会使已渲染好受到影响的部分失效, ...
- Python 元组(Tuple)操作详解
Python的元组与列表类似,不同之处在于元组的元素不能修改,元组使用小括号, 列表使用方括号,元组创建很简单,只需要在括号中添加元素,并使用逗号隔开即可 一.创建元组 代码如下: tup1 = (' ...
- e课表项目第二次冲刺周期第九天
昨天完成了什么? 昨天,我查找了相关的资料,将数据库根据我们的课程信息进行了重新的设计,并将数据能够连上数据库,即在添加课程的界面,可以将添加的课程的信息,存储到数据库中,并且存储到课程表中,并注明是 ...
- e课表项目第二次冲刺周期第五天
昨天干了什么? 昨天,我们组商量了主界面的主要设计,然后我查找了相关的资料,将主界面改成了一个连接数据库,并将数据库中的数据进行显示.接着,又在网上搜了一些资料,为数据建立了一个布局,能够显示该数据的 ...
- JavaScript七宗罪和一些槽点
当下JavaScript越来越流行,成为长期霸语言榜前三的语言.但是实际上JavaScript是一个很丑陋有很多槽点的语言,这就是为什么新出了那么多框架(从jQuery到Vue)以及海尔斯伯格大大推出 ...