给websocket加入心跳包防止自动断开连接_php websocket心跳包-程序员宅基地

技术标签: websocket  

var userId = $("#userId").val();
var lockReconnect = false; //避免ws重复连接
var ws = null; //判断当前浏览器是否支持WebSocket
var wsUrl = serverConfig.cyberhouse_ws+userId;
createWebSocket(wsUrl);   //连接ws

function createWebSocket(url) {
    try {
        if ('WebSocket' in window) {
            ws = new WebSocket(url);
        } else if ('MozWebSocket' in window) {
            ws = new MozWebSocket(url);
        } else {
            layui.use(['layer'], function() {
                var layer = layui.layer;
                layer.alert("您的浏览器不支持websocket的协议,建议使用新版谷歌、火狐等浏览器,请勿使用IE10以下浏览器,360浏览器请使用极速模式,不要使用兼容模式!");
            });
        }
        initEventHandle();
    } catch(e) {
        reconnect(url);
        console.log(e);
    }
}

function initEventHandle() {
    ws.onclose = function() {
        reconnect(wsUrl);
        console.log("llws连接关闭!"+new Data().toUTCString());
    };
    ws.onerror = function() {
        reconnect(wsUrl);
        console.log("llws连接错误!");
    };
    ws.onopen = function() {
        heartCheck.reset().start(); //心跳检测重置
        console.log("llws连接成功!"+new Data().toUTCString());
    };
    ws.onmessage = function(event) {//如果获得到消息、心跳检测重置
        heartCheck.reset().start();    //拿到任何消息都说明当前连接是正常的
        console.log("llws收到消息啦:"+event.data);
        if (event.data != 'pong') {
            var obj = eval("("+event.data+")");
            layui.use(['layim'], function(layim) {
                if (obj.type == "onlineStatus") {
                    layim.setFriendStatus(obj.id, obj.content);
                } else if (obj.type == "friend" || obj.type == "group") {
                    layim.getMessage(obj);
                }
            });
        }
    };
}

//监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server断会抛异常
window.onbeforeunload = function() {
    ws.close();
}

function reconnect(url) {
    if (lockReconnect) return;
    lockReconnect = true;
    setTimeout(function() { //没连接上会一直重连,设置延迟避免请求过多
        createWebSocket(url);
        lockReconnect = false;
    }, 2000);
}

//心跳检测
var heartCheck = {
    timeout: 540000,   //9分钟发一次心跳
    timeoutObj: null,
    serverTimeoutObj: null,
    reset: function() {
        clearTimeout(this.timeoutObj);
        clearTimeout(this.serverTimeoutObj);
        return this;
    },
    start: function() {
        var self = this;
        this.timeoutObj = setTimeout(function() {
            //这里发送一个心跳,后端收到后,返回一个心跳消息,
            //onmessage拿到返回的心跳就说明连接正常
            ws.send("ping");
            console.log("ping!");
            self.serverTimeoutObj = setTimeout(function() {//如果超过一定时间还没重置,说明后端主动断开了
                ws.close(); //如果onclose会执行reconnect,我们执行ws.close()就行了,如果直接执行reconnect会触发onclose导致重连两次
            }, self.timeout)
        }, this.timeout)
    }
}
//收到客户端消息后调用的方法
@OnMessage
public void onMessage(String message, Session session) {
    if (message.equals("ping")) {
    } else {
    ...
    }
}

系统发现websocket每隔10分钟自动断开连接,搜了很多博客都说设置一下nignx的

keepalive_timeout

proxy_send_timeout

proxy_connect_timeout

proxy_read_timeout

这四个字段的时长即可,然而好像并不生效。遂采取心跳包的方式每隔9分钟客户端自动发送ping消息给服务端,服务端不需要返回。即可解决问题。

转自:https://blog.csdn.net/jkxqj/article/details/77848466

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/zhangge3663/article/details/89294264

智能推荐

解决方案:mapper.xml文件报Invalid bound statement (not found)异常_invalid bound statement mapper.xml文件在resource里-程序员宅基地

文章浏览阅读938次。mapper.xml文件报Invalid bound statement not found 异常  如果在使用mapper.xml时报了Invalid bound statement (not found)的错误,并且自己的mapper.xml的路径和文件名都没有错误的情况,很可能是因为没有在pom.xml中配置如下代码: <!-- 如果不添加此节点mybatis的ma..._invalid bound statement mapper.xml文件在resource里

关于Spring Cloud Gateway网关出现大量synchronoss-file-upload-1000031191618557455空文件夹的问题-程序员宅基地

文章浏览阅读3k次,点赞6次,收藏4次。关于Spring Cloud Gateway网关出现大量synchronoss-file-upload-1000031191618557455空文件夹的问题_synchronoss-file-upload

ORACLE查询归档日志的切换频率及大小SQL-程序员宅基地

文章浏览阅读1.2k次。啥都不说了,见下面的语句: 点击(此处)折叠或打开 首先要确认一下每个归档日志(redo日志)文件的大小,通过下面的SQL: ..._oracle日志切频率 多少正常

Apache 官宣!Sharding-Sphere 首个 Apache 版本发布-程序员宅基地

文章浏览阅读140次。开发四年只会写业务代码,分布式高并发都不会还做程序员? >>> 竹外桃花三两枝,春江水暖鸭先知。四月..._shardingjdbc4.0什么时候发布

usb接口定义 及颜色-程序员宅基地

文章浏览阅读2.3k次。USB接口定义 颜色一般的排列方式是:红白绿黑从左到右定义:红色-USB电源 标有-VCC、Power、5V、5VSB字样绿色-USB数据线(正)-DATA+、USBD+、PD+、USBDT+白色-USB数据线(负)-DATA-..._usb2.0定义颜色

awk bc命令 linux_Linux命令行中使用计算器的方法:用bc,calc,expr,gcalccmd,qalc命令-程序员宅基地

文章浏览阅读194次。如果你需要在Linux命令行中使用计算器,那就按本文介绍的方法操作,用bc、calc、expr、gcalccmd、qalc命令,有些需要安装就可以用来去处了,这些命令行计算器允许我们执行所有类型的操作,例如科学、财务或甚至简单的计算,此外,我们可以在shell脚本中使用这些命令进行复杂的数学运算。简介在本文中,我列出几个命令行计算器命令:1、bc:任意精度计算器语言2、calc:任意精度计算器3、..._linux bc计算器awk

随便推点

阿里云服务器远程登录闪退_阿里云远程黑屏闪退怎么办-程序员宅基地

文章浏览阅读3.2k次。我的远程主机是1核1G内存配置这是远程主机的任务管理器截图: 以下是阿里技术交流结果: 最后,重启一下远程主机就OK啦FR:徐海涛(hunk Xu)_阿里云远程黑屏闪退怎么办

SpringBoot构建电商秒杀项目(二) 用户模块开发_根据页面传来的用户id,调用userservice的相关方法,获得user对象。-程序员宅基地

文章浏览阅读830次。第二章2.1 项目整体框架本项目采用springBoot开发,项目架构是Spring MVC框架controller层:与前端页面UI交互的层viewobject层:视图,封装了向前端展示的数据,避免领域模型的某些敏感数据直接传给前端dao层:与底层数据库交互的层,进行数据库的增删改查dataobject层:数据库表对于Java实体类的映射,是数据库表中数据的封装model层:领域..._根据页面传来的用户id,调用userservice的相关方法,获得user对象。

poj 1961 Period(KMP, 最短循环节)_给定一个字符串s,求最短循环节 java实现-程序员宅基地

文章浏览阅读1.7k次。题目链接:POJ : http://poj.org/problem?id=1961HDU : http://acm.hdu.edu.cn/showproblem.php?pid=1358ZOJ : http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=2177题目大意:给定一个长度为n的字符串s,_给定一个字符串s,求最短循环节 java实现

完整版ggplot作图(包括趋势线、Pearson相关分析、分类画图)_pearson 关联分析可以做图吗-程序员宅基地

文章浏览阅读249次。【代码】完整版ggplot作图(包括趋势线、Pearson相关分析、分类画图)_pearson 关联分析可以做图吗

http://www.flyaka.com/web.html4251-程序员宅基地

文章浏览阅读989次。I suspect that Kitty and Tom already have These materials are matched with three-dimensional flowers, crystal, b...

Android 左右滑屏,实现Back按键_android 模拟back按键事件-程序员宅基地

文章浏览阅读2.5k次。右滑屏,实现Back按键思路: 1.下拉快捷弹框如何实现,input事件处理 2.back按钮事件接口调用下拉分发事件代码路径:/frameworks/base/services/core/java/com/android/server/wm/WindowManagerService.javaframeworks/base/services/core/java/com/android/server/wm/RootWindowContainer.javaframeworks/base/ser_android 模拟back按键事件

推荐文章

热门文章

相关标签