JavaScript实现HTML5游戏断线自动重连的方法


Posted in Javascript onSeptember 18, 2017

断线重连的需求一断线重连原理二游戏内自动重连不刷新三刷新游戏自动重连重连数据Locationreplace重置url重连四实际项目中处理重连机制最后的总结断线重连的需求

尤其是手机上,会因为网络的不稳定或者其他原因,导致用户的socket链接断开。这个时候如果直接让玩家退出游戏,重新登录,无疑是非常影响用户体验的事情。所以根据这个需求,就有需要程序来实现断线后自动重连回去的技术,是用户能够再次快速开始游戏进行战斗。

一、断线重连原理

原来其实很简单,就是在断线的时候,根据用户的点击(有些时间短的就不用点击,默认是自动重连回来),程序自动识别,是要刷新重新进入游戏还是帮用户自动重连。客户端会根据自动重连标记,帮用户自动做事情。比如自动登陆、选角色、进入场景、请求同步后台数据等等。

根据实现机制,大致可以分两种实现方法。主要是游戏内自动重连(不刷新)和刷新游戏自动重连,后面会详细讲两种实现机制,以及相关的利弊。

二、游戏内自动重连(不刷新)

这种是比较难的,因为不刷新游戏,那么会因为一段时间的断开游戏,导致客户端数据跟服务器数据不同步了。比如怪物的位置、获得的奖励、进度等等。这些需要一开始就设计好,如果策划在后期要求加这个,那几乎是无法实现的,改动太大了。所以假设要这么多,大概是类似下面的做法。(如果真要详细,得一篇新的Blog了:)

1. 客户端和服务端协定好那些数据需要客户端自己同步

2. 断线多久内可以自动重连(策划以及技术上的实现来互相评估)

3. 服务端提供自动重连的协议,同时用户断线第一时间不应该就销毁掉相关数据(这里比较复杂,比如自动战斗是否要一直在服务器挂着,以及其他的一些关联操作)

4. 客户端不刷新游戏,使用新接口重新连接服务器,自动更新和同步相应的数据(比如同步怪物位置或者其他人物位置等等)

这种技术一般用于回合制之类的游戏,一般不涉及战斗系统。如果arpg的话,只能短暂的时间内可以自动重连,不然的话变数太大。或者需要做一些变种,比如单纯场景的怪物之类的刷新下,但是世界boss之类的场景就得重新载入之类的特殊处理。

三、刷新游戏自动重连

我个人感觉这个是简单粗暴又实用的做法。大部分游戏都适合使用,只要一刷新,游戏的数据都没了,全部重新开始,客户端只需要根据标记来做一些自动化的操作,容易很多。,同时服务器也不用更改,稳定也不容易出错。唯一不好的就是用户体验会稍微差一些。

重连数据

字符串数据:

//ip + 用户标识 + 服id + 服名字 (数据根据自己项目情况)
var reload:string = ip + "#" + token + "#" + serverId + "#" +serverName;
//后面的reload字符串都是这里的内容

重连标识:

reload //字符串

注意:刷新只能刷新自己的页面。(比如在iframe里面的时候)

Location.replace重置url重连

这种比较简单,也不会有什么兼容性的问题。就是重连的时候,把之前的登录用户以及服务端地址给记录起来

通过url来附带参数,最后再实际使用中解析出来,通过判断属性是否重载,

Location 对象的replace()方法:用新的文档替换当前文档。

通过传入新的url(其实是原url + 附带重连数据)

location.replace(oldUrl + "reload#" + reload);

是必须走url,而且还需要和原来的参数进行兼容处理。在游戏游戏中可能是这样的url了:

地调试1

http://localhost:63342/game/index.html?reload=1&host=ws://192.168.0.10:1050/ws&token

cookie 是存储于访问者的计算机中的变量。每当同一台计算机通过浏览器请求某个页面时,就会发送这个 cookie。你可以使用 JavaScript 来创建和取回 cookie 的值。

利用cookie的本地存放功能,也比较方便,但是会有些手机浏览器可能会无法读取到。

document.cookie = "reload#" + reload;

1.使用Html5的window.localStorage

localStorage 属性允许你访问一个 local Storage 对象。localStorage 与 sessionStorage 相似。不同之处在于,存储在 localStorage 里面的数据没有过期时间(expiration time),而存储在 sessionStorage 里面的数据会在浏览器会话(browsing session)结束时被清除,即浏览器关闭时。

应该注意的是,无论是 localStorage 还是 sessionStorage 中保存的数据都仅限于该页面的协议。

既然是H5游戏,当然重点是使用这个了,先看下localStorage的API

window.localStorage.setItem("reload", reload);

读取:

var reload = window.localStorage.getItem("reload");

1.使用Egret的本地保存

使用Egret会更方便,他进行了封装,H5和打包成本地都可以支持。localStorage.ts

egret.localStorage接口
//保存数据
export let getItem:(key:string)=>string;
//删除数据
export let removeItem:(key:string)=>void;
//将所有数据清空
export let clear:()=>void;
Web实现类WebLocalStorage.ts
namespace egret.localStorage.web {
function getItem(key:string):string {
return window.localStorage.getItem(key);
}
function setItem(key:string, value:string):boolean {
try{
window.localStorage.setItem(key, value);
return true;
}
catch(e){
egret.$warn(1047, key, value);
return false;
}
}
function removeItem(key:string):void {
window.localStorage.removeItem(key);
}
function clear():void {
window.localStorage.clear();
}
localStorage.getItem = getItem;
localStorage.setItem = setItem;
localStorage.removeItem = removeItem;
localStorage.clear = clear;
}

可以看到内部其实也是采用了window.localStorage来进行实现,同时做了一场处理,最后是通过localStorage接口进行赋值给外部调用。下面是实际使用方法:

//使用egret的本地存放方法做
egret.localStorage.setItem("reload",reload);
//游戏中获取
var reload:string = egret.localStorage.getItem("reload");

四、实际项目中处理重连机制

这里的代码是刷新之后重新进入游戏,然后通过之前的数据(url或者本地缓存)解析出相应的数据来进行判断。

解析url

var reload:string = location.href;

本地缓存获取数据

var reload:string = egret.localStorage.getItem("reload");
console.info("reload数据:" + reload);
if(reload && reload != "")
{
var cooks:string[] = reload.split("#");
console.info("断线重连刷新过来的");
this.session.isReload = true;
this.session.host = cooks[0];
this.session.token = cooks[1];
this.session.serverId = Number(cooks[2]);
this.session.serverName = cooks[3];
}

通过游戏内的变量进行判断处理

if(this.session.isReload)
{
//开始自动重连,走额外的协议以及自动处理
}
else
{
//走正常的流程
}

最后的总结

断线重连这里主要是讲了一些思路以及实际项目中的应用。代码算是伪5代码了,读者应该根据自己项目实际情况来设计,做一些相应变化,原理是一样的。

我们有一个项目是有遇到过其中的一个问题,因为一开始没有考虑自动重连的问题,在客户端和服务器都没这里的考虑。所以没办法做到不刷新游戏来进行重连(主要是成本太大,又是arpg游戏),最终是选择了游戏自己刷新来实现的机制的。当然也会遇到一些坑,比如接入一些平台,只能刷新自己的html,无法刷新平台的html(游戏内嵌),导致平台的sdk的问题(影响充值、关注等等)。不过最终都是有通过变通之类的进行解决了。

以上所述是小编给大家介绍的JavaScript实现HTML5游戏断线自动重连的方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
javascript学习网址备忘
May 29 Javascript
js操作label给label赋值及取label的值示例
Nov 07 Javascript
jQuery瀑布流插件Wookmark使用实例
Apr 02 Javascript
举例讲解JavaScript中将数组元素转换为字符串的方法
Oct 25 Javascript
基于jquery实现的鼠标悬停提示案例
Dec 11 Javascript
weex slider实现滑动底部导航功能
Aug 28 Javascript
ES6中Array.includes()函数的用法
Sep 20 Javascript
Js中将Long转换成日期格式的实现方法
Jun 05 Javascript
JS实现点击拉拽轮播图pc端移动端适配
Sep 05 Javascript
JavaScript面向对象编程小游戏---贪吃蛇代码实例
May 15 Javascript
JS实现音乐导航特效
Jan 06 Javascript
vue自动添加浏览器兼容前后缀操作
Aug 13 Javascript
Node.JS中快速扫描端口并发现局域网内的Web服务器地址(80)
Sep 18 #Javascript
BetterScroll 在移动端滚动场景的应用
Sep 18 #Javascript
Windows下Node.js安装及环境配置方法
Sep 18 #Javascript
jQuery选择器之子元素选择器详解
Sep 18 #jQuery
JavaScript实现的斑马线表格效果【隔行变色】
Sep 18 #Javascript
js实现可以点击收缩或张开的悬浮窗
Sep 18 #Javascript
javascript流程控制语句集合
Sep 18 #Javascript
You might like
使用Linux五年积累的一些经验技巧
2013/06/20 PHP
PHP实现读取一个1G的文件大小
2013/08/24 PHP
从零开始学YII2框架(二)通过 Composer 安装扩展插件
2014/08/20 PHP
php中array_multisort对多维数组排序的方法
2020/06/21 PHP
Jquery 常用方法经典总结
2010/01/28 Javascript
jQuery 源码分析笔记(2) 变量列表
2011/05/28 Javascript
70+漂亮且极具亲和力的导航菜单设计国外网站推荐
2011/09/20 Javascript
js两行代码按指定格式输出日期时间
2011/10/21 Javascript
javascript通过navigator.userAgent识别各种浏览器
2013/10/25 Javascript
父节点获取子节点的字符串示例代码
2014/02/26 Javascript
解决wx.onMenuShareTimeline出现的问题
2016/08/16 Javascript
jQuery基于xml格式数据实现模糊查询及分页功能的方法
2016/12/25 Javascript
微信小程序实现表单校验功能
2020/03/30 Javascript
分析javascript原型及原型链
2018/03/18 Javascript
基于Angularjs-router动态改变Title值的问题
2018/08/30 Javascript
微信小程序 调用微信授权窗口相关问题解决
2019/07/25 Javascript
[01:03:54]Liquid vs IG 2018国际邀请赛小组赛BO2 第一场 8.17
2018/08/18 DOTA
python控制台显示时钟的示例
2014/02/24 Python
详解详解Python中writelines()方法的使用
2015/05/25 Python
让Python代码更快运行的5种方法
2015/06/21 Python
python函数与方法的区别总结
2019/06/23 Python
调用其他python脚本文件里面的类和方法过程解析
2019/11/15 Python
python支持多线程的爬虫实例
2019/12/21 Python
pytorch 查看cuda 版本方式
2020/06/23 Python
CSS3弹性盒模型开发笔记(一)
2016/04/26 HTML / CSS
使用CSS3的font-face字体嵌入样式的方法讲解
2016/05/13 HTML / CSS
香港时装购物网站:ZALORA香港
2017/04/23 全球购物
世界上最好的足球商店:Unisport
2019/03/02 全球购物
英国钻石公司:British Diamond Company
2020/02/16 全球购物
室内设计专业个人的自我评价
2013/10/19 职场文书
环境科学专业个人求职的自我评价
2013/11/28 职场文书
2014年劳动部工作总结
2014/12/11 职场文书
2015年保险公司内勤工作总结
2015/05/23 职场文书
科级干部培训心得体会
2016/01/06 职场文书
环境保护宣传标语大全!
2019/06/28 职场文书
vue-cropper插件实现图片截取上传组件封装
2021/05/27 Vue.js