范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文

一文吃透WebSocket原理

  作者:Gaby 原文链接 https://juejin.cn/post/7020964728386093093一.前言
  踩着年末的尾巴,提前布局来年,为来年的工作做个好的铺垫,所以就开始了面试历程,因为项目中使用到了 WebSocket  ,面试官在深挖项目经验的时候,也难免提到 WebSocket  相关的知识点,因为之前并没有考虑这么深,所以,回答的还是有所欠缺,因此,赶紧趁热再熟悉熟悉,也借此机会,整理出来供大家咀嚼,每个项目都有其值得挖掘的闪光点,要用有爱的眼睛去发现。二.什么是 WebSocket
  WebSocket  是一种在单个TCP连接上进行全双工通信的协议。WebSocket  使得客户端和服务器之间的数据交换变得更加简单,允许服务端主动向客户端推送数据。
  在 WebSocket API  中,浏览器和服务器只需要完成一次握手,两者之间就直接可以创建持久性的连接, 并进行双向数据传输。(维基百科)
  WebSocket  本质上一种计算机网络应用层的协议,用来弥补 http  协议在持久通信能力上的不足。
  WebSocket 协议在2008年诞生,2011年成为国际标准。现在最新版本浏览器都已经支持了。
  它的最大特点就是,服务器可以主动向客户端推送信息,客户端也可以主动向服务器发送信息,是真正的双向平等对话,属于服务器推送技术的一种。
  WebSocket 的其他特点包括:(1)建立在 TCP 协议之上,服务器端的实现比较容易。 (2)与 HTTP 协议有着良好的兼容性。默认端口也是80和443,并且握手阶段采用 HTTP 协议,因此握手时不容易屏蔽,能通过各种 HTTP 代理服务器。 (3)数据格式比较轻量,性能开销小,通信高效。 (4)可以发送文本,也可以发送二进制数据。 (5)没有同源限制,客户端可以与任意服务器通信。 (6)协议标识符是ws(如果加密,则为wss),服务器网址就是 URL。 ws://example.com:80/some/path
  为什么需要 WebSocket?
  我们已经有了 HTTP 协议,为什么还需要另一个协议?它能带来什么好处?
  因为 HTTP 协议有一个缺陷:通信只能由客户端发起,不具备服务器推送能力。
  举例来说,我们想了解查询今天的实时数据,只能是客户端向服务器发出请求,服务器返回查询结果。HTTP 协议做不到服务器主动向客户端推送信息。
  这种单向请求的特点,注定了如果服务器有连续的状态变化,客户端要获知就非常麻烦。我们只能使用"轮询":每隔一段时候,就发出一个询问,了解服务器有没有新的信息。最典型的场景就是聊天室。轮询的效率低,非常浪费资源(因为必须不停连接,或者 HTTP 连接始终打开)。
  在 WebSocket  协议出现以前,创建一个和服务端进双通道通信的 web 应用,需要依赖HTTP协议,进行不停的轮询,这会导致一些问题:服务端被迫维持来自每个客户端的大量不同的连接 大量的轮询请求会造成高开销,比如会带上多余的header,造成了无用的数据传输。
  http  协议本身是没有持久通信能力的,但是我们在实际的应用中,是很需要这种能力的,所以,为了解决这些问题,WebSocket  协议由此而生,于2011年被IETF定为标准RFC6455,并被RFC7936所补充规范。并且在 HTML5  标准中增加了有关 WebSocket  协议的相关 api  ,所以只要实现了 HTML5  标准的客户端,就可以与支持 WebSocket  协议的服务器进行全双工的持久通信了。WebSocket 与 HTTP 的区别
  WebSocket  与 HTTP  的关系图:
  相同点:  都是一样基于TCP的,都是可靠性传输协议。都是应用层协议。 联系:  WebSocket在建立握手时,数据是通过HTTP传输的。但是建立之后,在真正传输时候是不需要HTTP协议的。
  下面一张图说明了 HTTP 与 WebSocket 的主要区别:
  不同点:1、  WebSocket  是双向通信协议,模拟 Socket  协议,可以双向发送或接受信息,而 HTTP  是单向的;2、  WebSocket  是需要浏览器和服务器握手进行建立连接的,而 http  是浏览器发起向服务器的连接。3、 虽然  HTTP/2  也具备服务器推送功能,但 HTTP/2  只能推送静态资源,无法推送指定的信息。三、WebSocket协议的原理
  与http协议一样, WebSocket  协议也需要通过已建立的TCP连接来传输数据。具体实现上是通过http协议建立通道,然后在此基础上用真正 WebSocket  协议进行通信,所以WebSocket协议和http协议是有一定的交叉关系的。首先, WebSocket  是一个持久化的协议,相对于 HTTP 这种非持久的协议来说。简单的举个例子吧,用目前应用比较广泛的 PHP 生命周期来解释。
  HTTP 的生命周期通过 Request 来界定,也就是一个 Request 一个 Response ,那么在 HTTP1.0 中,这次 HTTP 请求就结束了。
  在 HTTP1.1 中进行了改进,使得有一个 keep-alive,也就是说,在一个 HTTP 连接中,可以发送多个 Request,接收多个 Response。但是请记住 Request = Response, 在 HTTP 中永远是这样,也就是说一个 Request 只能有一个 Response。而且这个 Response 也是被动的,不能主动发起。首先 WebSocket 是基于 HTTP 协议的,或者说借用了 HTTP 协议来完成一部分握手。
  首先我们来看个典型的 WebSocket 握手GET /chat HTTP/1.1 Host: server.example.com Upgrade: websocket Connection: Upgrade Sec-WebSocket-Key: x3JJHMbDL1EzLkh9GBhXDw== Sec-WebSocket-Protocol: chat, superchat Sec-WebSocket-Version: 13 Origin: http://example.com
  熟悉 HTTP 的童鞋可能发现了,这段类似 HTTP 协议的握手请求中,多了这么几个东西。Upgrade: websocket Connection: Upgrade
  这个就是 WebSocket  的核心了,告诉 Apache 、 Nginx  等服务器:注意啦,我发起的请求要用 WebSocket  协议,快点帮我找到对应的助理处理~而不是那个老土的 HTTP  。Sec-WebSocket-Key: x3JJHMbDL1EzLkh9GBhXDw== Sec-WebSocket-Protocol: chat, superchat Sec-WebSocket-Version: 13 首先, Sec-WebSocket-Key 是一个 Base64 encode 的值,这个是浏览器随机生成的,告诉服务器:泥煤,不要忽悠我,我要验证你是不是真的是 WebSocket 助理。 然后, Sec_WebSocket-Protocol 是一个用户定义的字符串,用来区分同 URL 下,不同的服务所需要的协议。简单理解:今晚我要服务A,别搞错啦~ 最后, Sec-WebSocket-Version 是告诉服务器所使用的 WebSocket Draft (协议版本),在最初的时候,WebSocket 协议还在 Draft 阶段,各种奇奇怪怪的协议都有,而且还有很多期奇奇怪怪不同的东西,什么 Firefox 和 Chrome 用的不是一个版本之类的,当初 WebSocket 协议太多可是一个大难题…不过现在还好,已经定下来啦~大家都使用同一个版本:服务员,我要的是13岁的噢 _  然后服务器会返回下列东西,表示已经接受到请求, 成功建立 WebSocket 啦! HTTP/1.1 101 Switching Protocols Upgrade: websocket Connection: Upgrade Sec-WebSocket-Accept: HSmrc0sMlYUkAGmm5OPpG2HaGWk= Sec-WebSocket-Protocol: chat
  这里开始就是 HTTP 最后负责的区域了,告诉客户,我已经成功切换协议啦~Upgrade: websocket Connection: Upgrade
  依然是固定的,告诉客户端即将升级的是 WebSocket 协议,而不是 mozillasocket  ,lurnarsocket  或者 shitsocket 。
  然后, Sec-WebSocket-Accept  这个则是经过服务器确认,并且加密过后的 Sec-WebSocket-Key 。服务器:好啦好啦,知道啦,给你看我的 ID CARD 来证明行了吧。后面的, Sec-WebSocket-Protocol  则是表示最终使用的协议。至此,HTTP 已经完成它所有工作了,接下来就是完全按照 WebSocket  协议进行了。总结, WebSocket  连接的过程是:首先,客户端发起http请求,经过3次握手后,建立起TCP连接; http  请求里存放 WebSocket  支持的版本号等信息,如:Upgrade、Connection、WebSocket-Version等;然后,服务器收到客户端的握手请求后,同样采用HTTP协议回馈数据; 最后,客户端收到连接成功的消息后,开始借助于TCP传输信道进行全双工通信。 四、Websocket的优缺点
  优点:WebSocket协议一旦建议后,互相沟通所消耗的请求头是很小的 服务器可以向客户端推送消息了
  缺点:少部分浏览器不支持,浏览器支持的程度与方式有区别(IE10) 五、WebSocket应用场景即时聊天通信 多玩家游戏 在线协同编辑/编辑 实时数据流的拉取与推送 体育/游戏实况 实时地图位置 即时Web应用程序:即时Web应用程序使用一个Web套接字在客户端显示数据,这些数据由后端服务器连续发送。在WebSocket中,数据被连续推送/传输到已经打开的同一连接中,这就是为什么WebSocket更快并提高了应用程序性能的原因。例如在交易网站或比特币交易中,这是最不稳定的事情,它用于显示价格波动,数据被后端服务器使用Web套接字通道连续推送到客户端。 游戏应用程序:在游戏应用程序中,你可能会注意到,服务器会持续接收数据,而不会刷新用户界面。屏幕上的用户界面会自动刷新,而且不需要建立新的连接,因此在WebSocket游戏应用程序中非常有帮助。 聊天应用程序:聊天应用程序仅使用WebSocket建立一次连接,便能在订阅户之间交换,发布和广播消息。它重复使用相同的WebSocket连接,用于发送和接收消息以及一对一的消息传输。 六、websocket 断线重连
  心跳就是客户端定时的给服务端发送消息,证明客户端是在线的, 如果超过一定的时间没有发送则就是离线了。如何判断在线离线?
  当客户端第一次发送请求至服务端时会携带唯一标识、以及时间戳,服务端到db或者缓存去查询改请求的唯一标识,如果不存在就存入db或者缓存中, 第二次客户端定时再次发送请求依旧携带唯一标识、以及时间戳,服务端到db或者缓存去查询改请求的唯一标识,如果存在就把上次的时间戳拿取出来,使用当前时间戳减去上次的时间, 得出的毫秒秒数判断是否大于指定的时间,若小于的话就是在线,否则就是离线;如何解决断线问题
  通过查阅资料了解到 nginx 代理的 websocket 转发,无消息连接会出现超时断开问题。网上资料提到解决方案两种,一种是修改nginx配置信息,第二种是 websocket  发送心跳包。下面就来总结一下本次项目实践中解决的 websocket  的断线 和 重连 这两个问题的解决方案。主动触发包括主动断开连接,客户端主动发送消息给后端1 主动断开连接 ws.close();
  主动断开连接,根据需要使用,基本很少用到。2 主动发送消息 ws.send("hello world"); 断线的可能原因1:websocket超时没有消息自动断开连接,应对措施:这时候我们就需要知道服务端设置的超时时长是多少,在小于超时时间内发送心跳包,有2中方案:一种是客户端主动发送上行心跳包,另一种方案是服务端主动发送下行心跳包。
  下面主要讲一下客户端也就是前端如何实现心跳包:
  首先了解一下心跳包机制
  跳包之所以叫心跳包是因为:它像心跳一样每隔固定时间发一次,以此来告诉服务器,这个客户端还活着。事实上这是为了保持长连接,至于这个包的内容,是没有什么特别规定的,不过一般都是很小的包,或者只包含包头的一个空包。
  在 TCP  的机制里面,本身是存在有心跳包的机制的,也就是 TCP  的选项:SO_KEEPALIVE  。系统默认是设置的2小时的心跳频率。但是它检查不到机器断电、网线拔出、防火墙这些断线。而且逻辑层处理断线可能也不是那么好处理。一般,如果只是用于保活还是可以的。
  心跳包一般来说都是在逻辑层发送空的 echo  包来实现的。下一个定时器,在一定时间间隔下发送一个空包给客户端,然后客户端反馈一个同样的空包回来,服务器如果在一定时间内收不到客户端发送过来的反馈包,那就只有认定说掉线了。
  在长连接下,有可能很长一段时间都没有数据往来。理论上说,这个连接是一直保持连接的,但是实际情况中,如果中间节点出现什么故障是难以知道的。更要命的是,有的节点(防火墙)会自动把一定时间之内没有数据交互的连接给断掉。在这个时候,就需要我们的心跳包了,用于维持长连接,保活。
  心跳检测步骤:客户端每隔一个时间间隔发生一个探测包给服务器 客户端发包时启动一个超时定时器 服务器端接收到检测包,应该回应一个包 如果客户机收到服务器的应答包,则说明服务器正常,删除超时定时器 如果客户端的超时定时器超时,依然没有收到应答包,则说明服务器挂了 // 前端解决方案:心跳检测 var heartCheck = {     timeout: 30000, //30秒发一次心跳     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);     } }  断线的可能原因2: websocket  异常包括服务端出现中断,交互切屏等等客户端异常中断等等 当若服务端宕机了,客户端怎么做、服务端再次上线时怎么做?客户端则需要断开连接,通过 onclose  关闭连接,服务端再次上线时则需要清除之间存的数据,若不清除 则会造成只要请求到服务端的都会被视为离线。
  针对这种异常的中断解决方案就是处理重连,下面我们给出的重连方案是使用js库处理:引入reconnecting-websocket.min.js ,ws建立链接方法使用js库api方法:var ws = new ReconnectingWebSocket(url); // 断线重连: reconnectSocket(){     if ("ws" in window) {         ws = new ReconnectingWebSocket(url);     } else if ("MozWebSocket" in window) {        ws = new MozWebSocket(url);     } else {       ws = new SockJS(url);     } }
  断网监测支持使用js库:offline.min.js onLineCheck(){     Offline.check();     console.log(Offline.state,"---Offline.state");     console.log(this.socketStatus,"---this.socketStatus");      if(!this.socketStatus){         console.log("网络连接已断开!");         if(Offline.state === "up" && websocket.reconnectAttempts > websocket.maxReconnectInterval){             window.location.reload();         }         reconnectSocket();     }else{         console.log("网络连接成功!");         websocket.send("heartBeat");     } }  // 使用:在websocket断开链接时调用网络中断监测 websocket.onclose => () {     onLineCheck(); };
  以上方案,只是抛砖引玉,如果大家有更好的解决方案欢迎评论区分享交流。七、总结
  WebSocket 是为了在 web 应用上进行双通道通信而产生的协议,相比于轮询HTTP请求的方式,WebSocket 有节省服务器资源,效率高等优点。WebSocket 中的掩码是为了防止早期版本中存在中间缓存污染攻击等问题而设置的,客户端向服务端发送数据需要掩码,服务端向客户端发送数据不需要掩码。WebSocket 中 Sec-WebSocket-Key 的生成算法是拼接服务端和客户端生成的字符串,进行SHA1哈希算法,再用base64编码。WebSocket 协议握手是依靠 HTTP 协议的,依靠于 HTTP 响应101进行协议升级转换。参考阮一峰:WebSocket 教程 看完让你彻底理解 WebSocket 原理

国美与腾讯签署合作协议助力实体门店数字化潘宇静中国证券报中证网中证网讯(记者潘宇静)线上线下融合再添新案例。5月5日,国美零售与腾讯控股签署战略合作协议,双方将共同打造数实融合的全零售样板。本次签约涵盖双方在大数据云互联管理层纷争有望落幕,安谋科技新任领导层亮相线上全员大会澎湃新闻记者周玲安谋科技(中国)有限公司(以下简称安谋科技)管理层变更纷争有望落下帷幕。5月6日,安谋科技召开多地办公室员工线上全员大会,公司新任联席首席执行官刘仁辰与陈恂集体亮相联通八元保号套餐办理基本靠怼和威胁,不用客气。我改8元套餐。客服线上改不了。我工信部说能改。客服要登记申请。我非得投诉吗?客服48h,今天月底改不了。我xxxxx是工号对吧。客服为您提交上去,等专员电获KPL认证的手游利器!游戏玩家同样值得关注vivoX80在上个月刚刚发布的vivoX80,其拔尖的影像表现,给许多用户都留下了深刻印象。作为一款旗舰手机,vivoX80的表现自然也是足够全面的。这一次,新机带来了升级版旗舰双芯片,viv游戏竞技体验全面突围,红魔变形金刚限量定制版不止制服原神随着生活方式的转变,我们的娱乐消遣方式出现了显著变化,在生活的间隙,越来越多的人开始选择玩游戏来缓解自己的工作和生活压力。为此,国内各大手机厂商业也都纷纷转变手机的优化思路,从之前网易加速游戏业务出海,于德克萨斯州开设首家美国工作室记者彭新编辑发力海外游戏业务,成为网易游戏业务未来的重点。5月5日,网易游戏宣布,旗下JackalopeGames于美国德克萨斯州奥斯汀成立,成为其首个北美工作室。新工作室由资深游硬刚!先罚谷歌540亿,又罚苹果1020亿,特朗普没人比她恨美国硬刚!先罚谷歌540亿,又罚苹果1020亿,特朗普没人比她恨美国我们都知道,如今的欧洲虽然也是发达国家,但是在很多的领域是高度依赖美国的,甚至几乎没有互联网巨头,在这个信息化的时代哪来的钱发给自媒体?头条能给创作者发工资,就说明头条是盈利的,或者说建立之初虽然不盈利,但也是鼓励创作者能够分享丰富头条内容。今日头条是国内为数不多的百亿级媒体资讯平台,跟腾讯百度阿里等一样有着强大的微信右上角的号,还隐藏着4个你不知道的功能,不会用可惜了原来微信右上角的号这么强大!还隐藏着4个你不知道的功能,非常实用,不会用太浪费了!大家好,我是秦韵莞香,用微信这么久了,估计很多人都没有发现,原来微信右上角的号功能这么强大!这4个书评丨硅谷是如何起飞的?文郑磊从事风险投资的人都知道美国有两个创新技术发源地,也是风投资金扎堆的地方一个在东海岸的波士顿,一个在西海岸的硅谷。在美国,他们分别是在20世纪上半叶无线电技术和下半叶半导体芯片oracle表的crud(增删改查)单表增1增加数据insertinto表名(属性1,属性2,属性3。,)values(属性1对应的数据类型的值,属性2对应的数据类型的值,属性3对应的数据类型的值,)删1。删除一行数据d
同庆建军佳节共叙鱼水深情翼联EDUP欢度八一建军节2021年8月1日,是中国人民解放军建军94周年纪念日。7月30日下午,翼联EDUP以同庆建军佳节共叙鱼水深情为主题,举办了热烈欢快的建军节活动。智力比拼建军知识竞答全体小伙伴认真有无WiWD9902无线投屏器趣意来袭打造一个家庭影院已变得愈加简单,使用无线投屏器,手机4K电影开启,电视同步播放,超清大屏更沉浸。至于举办一场游戏比赛就更加容易了。只需连接一台参赛手机,采用无线投屏器,将画面同屏到又省又快免装宽带翼联EDUP4G随身wifiD623S嗨翻今夏虽说如今的宽带已进入寻常百姓家,但作为打工人学生党,上网依然不是一件容易事。随身wifi已经出来好些年头了,好用易用动静相宜的随身wifi却是迷之存在。接下来,为大家介绍一款翼联E干杯翼联EDUP助威蜜雪冰城致敬火热青春7月26日下午,翼联EDUP开设了一个特别的下午茶环节,董事长王承先生为每一位小伙伴都奉上了一杯清凉的蜜雪冰城饮品,号召大家支持国货,向有责任有担当的民族品牌看齐,坚定理想不懈努力5个摄像头15颗雷达全新明锐PRO12。49万起售距离上海车展首发一周后,上汽大众斯柯达全新明锐PRO于5月17日正式上市。新车推出4款车型,售价区间为12。49万15。79万元。全新明锐PRO是斯柯达品牌旗下首款采用全新MQBE痛惜新能源车频繁自燃,再看比亚迪的安全底线01hr今年以来,多款新能源车频发自燃现象,伴随的剧烈的爆炸更让人心惊。消费者在感受着新能源车带来的新奇体验同时,也充斥着紧张与不安。安全再次战胜理性,成为关注新能源车的第一首选。榜单SUV保值率TOP5透露车越大越保值近日,中国汽车流通协会和精真估联合发布了最新的保值率数据及研究报告。报告显示,在SUV领域,级别越高的车型保值率越高,在二手市场价值更高,而小型SUV的保值率相对较低且下滑趋势明显奔驰电动化提速,2030年燃油车数量减少七成汽车电动化已不可逆,发明汽车的奔驰不再犹豫,开始了急行军。至2030年,电动车将占新车销量的半数以上,传统燃油车数量将减少七成。上周,奔驰面向全球投资者抛出一份未来计划,包括AMG转转大数据OPPOReno成为上半年最保值的旗舰机作为电子产品中最为普及的一种,手机可以说是最不保值的,用入手即贬值来形容可以说是非常贴切了。除了厂商频繁更新硬件迭代新品造成的原因外,来自官方的促销降价等手段更使得手机保值成为了难硬件技术升级,比亚迪汉首搭博世IPB制动系统比亚迪汉是中国首个搭载博世IPB智能集成制动系统的新能源量产车,同时基于IPB系统的CST舒适制动功能在汉上实现全球首发。IPB系统为汉实现领先的安全性舒适性和长续航提供了帮助。I小猛士变学霸局座张召忠国手刘国梁为风神AX7PRO站台以一场极限挑战真人秀的形式,东风风神AX7PRO于9月10日在线上市。著名军事评论家张召忠中国乒乓球协会主席刘国梁为新车硬核实力站台,见证栋梁本色。猛士压顶十秒钟移动堡垒更可靠备质