菜单

HTML5的Websocket(理论篇 I)

2019年2月18日 - CSS/CSS3

HTML5的Websocket(理论篇 I)

2017/10/28 · HTML5 ·
websocket

原稿出处:
走走前端   

先请来TA的邻居:

http:无状态、基于tcp呼吁/响应格局的应用层磋商
(A:哎哎,上次你请自身吃饭了么? B:小编心想, 上次请你吃了么)
tcp:面向连接、保险高可信性(数据无遗失、数据无失序、数据无不当、数据无重复到达)
传输层协商。(看呀,大阅兵,如此规整有秩序)

为啥要引入Websocket:

TiggoFC开篇介绍:本协议的目标是为着消除基于浏览器的次第要求拉取财富时必须发起五个HTTP请求和长日子的轮询的题材。

long poll(长轮询):
客户端发送一个request后,服务器得到这么些两次三番,即使有消息,才重回response给客户端。没有音讯,就一向不回去response。之后客户端再度发送request,
重复上次的动作。

图片 1

从上得以看来,http协议的表征是服务器不能积极联系客户端,只好由客户端发起。它的被动性预示了在已毕双向通讯时必要不停的接连或一连一向打开,这就要求服务器快捷的处理速度或高并发的能力,是相当消功耗源的。

那几个时候,Websocket出现了。

原稿出处:
走走前端   

Websocket是什么:

奥德赛FC中写到:WebSocket协议使在支配环境下运作不受信任代码的客户端和力所能及挑选与那多少个代码通讯的远程主机之间可以双向通讯。

对,划重点:双向通信

Websocket在连年之后,客户端能够积极发送音讯给服务器,服务器也足以积极向客户端推送音信。比如:预定车票音讯,除了大家发请求询问车票怎样,当然更期待倘使有新音讯,可以一向文告大家。

其特点:

(1)握手阶段采用 HTTP 协议,默许端口是80和443

(2)建立在TCP协议基础之上,和http协议同属于应用层

(4)可以发送文书,也足以发送二进制数据

(5)没有同源限制,客户端可以与人身自由服务器通讯

(6)协议标识符是ws(若是加密,为wss),如ws://localhost:8023

简言之来说,Websocket协商分成两部分:握手和数量传输。

图片 2

先请来TA的邻居:

http:无状态、基于tcp呼吁/响应方式的应用层商量(A:哎哎,上次您请自身吃饭了么? B:笔者考虑, 上次请您吃了么)
tcp:面向连接、保障高可信赖性(数据无遗失、数据无失序、数据无不当、数据无重复到达)
传输层情商。(看呀,大阅兵,如此规整有秩序)

为什么要引入Websocket:

途乐FC开篇介绍:本协议的目的是为着化解基于浏览器的次第要求拉取能源时务必发起多少个HTTP请求和长日子的轮询的题材。

Websocket API:

那里是指客户端 API。

long poll(长轮询):
客户端发送2个request后,服务器得到那一个连续,假诺有音信,才回到response给客户端。没有音讯,就向来不回来response。之后客户端再一次发送request,
重复上次的动作。

WebSocket 构造函数

经过调用WebSocket构造函数来创制七个WebSocket实例对象,建立客户端与服务器的连日。

JavaScript

const ws = new WebSocket(‘ws://localhost:8023’);

1
const ws = new WebSocket(‘ws://localhost:8023’);

图片 3

Websocket事件

WebSocket 是纯事件驱动,通过监听事件可以拍卖到来的数目和更改的一连情状。服务端发送数据后,消息和事件会异步到达。

JavaScript

// 连接请求open事件处理: ws.onopen = e => {
console.log('Connection success'); ws.send(\`Hello ${e}\`); };

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5b8f447934b5b531196143-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b5b531196143-2">
2
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b5b531196143-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b5b531196143-4">
4
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b5b531196143-5">
5
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5b8f447934b5b531196143-1" class="crayon-line">
 // 连接请求open事件处理:
</div>
<div id="crayon-5b8f447934b5b531196143-2" class="crayon-line crayon-striped-line">
     ws.onopen = e =&gt; {
</div>
<div id="crayon-5b8f447934b5b531196143-3" class="crayon-line">
         console.log('Connection success');
</div>
<div id="crayon-5b8f447934b5b531196143-4" class="crayon-line crayon-striped-line">
         ws.send(`Hello ${e}`);
</div>
<div id="crayon-5b8f447934b5b531196143-5" class="crayon-line">
     };
</div>
</div></td>
</tr>
</tbody>
</table>

假诺要钦赐四个回调函数,能够采用add伊芙ntListener方法。

JavaScript

ws.addEventListener(‘open’, e => { ws.send(`Hello ${e}`); });

1
2
3
ws.addEventListener(‘open’, e => {
  ws.send(`Hello ${e}`);
});

当open事件触发时,意味着握手阶段已甘休。服务端已经处理了连接的乞求,可以准备收发数据。

JavaScript

// 接受文本消息的事件处理: ws.onmessage = e =&gt; { const data =
e.data; if (typeof data === "string") { console.log("Received string
message ",data); } else if (data instanceof Blob) {
console.log("Received blob message ", data); } };

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5b8f447934b62129912854-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b62129912854-2">
2
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b62129912854-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b62129912854-4">
4
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b62129912854-5">
5
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b62129912854-6">
6
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b62129912854-7">
7
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b62129912854-8">
8
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b62129912854-9">
9
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5b8f447934b62129912854-1" class="crayon-line">
// 接受文本消息的事件处理:
</div>
<div id="crayon-5b8f447934b62129912854-2" class="crayon-line crayon-striped-line">
ws.onmessage = e =&gt; {
</div>
<div id="crayon-5b8f447934b62129912854-3" class="crayon-line">
    const data = e.data;
</div>
<div id="crayon-5b8f447934b62129912854-4" class="crayon-line crayon-striped-line">
    if (typeof data === &quot;string&quot;) {
</div>
<div id="crayon-5b8f447934b62129912854-5" class="crayon-line">
        console.log(&quot;Received string message &quot;,data);
</div>
<div id="crayon-5b8f447934b62129912854-6" class="crayon-line crayon-striped-line">
    } else if (data instanceof Blob) {
</div>
<div id="crayon-5b8f447934b62129912854-7" class="crayon-line">
        console.log(&quot;Received blob message &quot;, data);
</div>
<div id="crayon-5b8f447934b62129912854-8" class="crayon-line crayon-striped-line">
    }
</div>
<div id="crayon-5b8f447934b62129912854-9" class="crayon-line">
};
</div>
</div></td>
</tr>
</tbody>
</table>

服务器数据可能是文本,也大概是二进制数据,有Blob和ArrayBuffer两连串型,在读取到多少以前要求控制好数据的品种。

JavaScript

//异常处理 ws.onerror = e =&gt; { console.log("WebSocket Error: " ,
e); handleErrors(e); };

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5b8f447934b66862080563-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b66862080563-2">
2
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b66862080563-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b66862080563-4">
4
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b66862080563-5">
5
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5b8f447934b66862080563-1" class="crayon-line">
//异常处理
</div>
<div id="crayon-5b8f447934b66862080563-2" class="crayon-line crayon-striped-line">
ws.onerror = e =&gt; {
</div>
<div id="crayon-5b8f447934b66862080563-3" class="crayon-line">
    console.log(&quot;WebSocket Error: &quot; , e);
</div>
<div id="crayon-5b8f447934b66862080563-4" class="crayon-line crayon-striped-line">
    handleErrors(e);
</div>
<div id="crayon-5b8f447934b66862080563-5" class="crayon-line">
};
</div>
</div></td>
</tr>
</tbody>
</table>

WebSocket 规范中定义了ping 帧 和pong
帧,可以用来做心跳重连,互联网状态查询等,不过当前
浏览器只会活动发送pong帧,而不会发ping 帧。(有趣味可详查ping和pong帧)

JavaScript

//关闭连接处理 ws.onclose = e => { const code = e.code; const reason
= e.reason; console.log(“Connection close”, code, reason); };

1
2
3
4
5
6
//关闭连接处理
ws.onclose = e => {
    const code = e.code;
    const reason = e.reason;
    console.log("Connection close", code, reason);
};

从上得以看看,http协议的性状是服务器无法积极联系客户端,只好由客户端发起。它的被动性预示了在形成双向通讯时必要不停的两次三番或一而再向来打开,那就须要服务器赶快的处理速度或高并发的能力,是格外消耗财富的。

WebSocket 方法:

WebSocket 对象有多少个措施:send 和 close

JavaScript

//发送一个文本消息 ws.send("this is websocket");

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5b8f447934b6d916593124-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b6d916593124-2">
2
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5b8f447934b6d916593124-1" class="crayon-line">
//发送一个文本消息
</div>
<div id="crayon-5b8f447934b6d916593124-2" class="crayon-line crayon-striped-line">
ws.send(&quot;this is websocket&quot;);
</div>
</div></td>
</tr>
</tbody>
</table>

在open事件的回调中调用send()方法传送数据:

JavaScript

const ws = new WebSocket(‘ws://localhost:8023’); ws.onopen = e => {
console.log(‘Connection success’); ws.send(`Hello ${e}`); };

1
2
3
4
5
const ws = new WebSocket(‘ws://localhost:8023’);
ws.onopen = e => {
    console.log(‘Connection success’);
    ws.send(`Hello ${e}`);
};

如若想经过响应其余事件发送音信,可因而判断当前的Websocket的readyState属性。接下来会说到readyState.

JavaScript

ws.close(); //1000是状态码,代表正常结束。 ws.close(1000, "Closing
normally");

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5b8f447934b73487491254-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b73487491254-2">
2
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b73487491254-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b73487491254-4">
4
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5b8f447934b73487491254-1" class="crayon-line">
ws.close();
</div>
<div id="crayon-5b8f447934b73487491254-2" class="crayon-line crayon-striped-line">
 
</div>
<div id="crayon-5b8f447934b73487491254-3" class="crayon-line">
//1000是状态码,代表正常结束。
</div>
<div id="crayon-5b8f447934b73487491254-4" class="crayon-line crayon-striped-line">
ws.close(1000, &quot;Closing normally&quot;);
</div>
</div></td>
</tr>
</tbody>
</table>

那一个时候,Websocket出现了。

WebSocket 属性

readyState值表示连接景况,是只读属性。它有以下两个值:

WebSocket.CONNECTING :连接正在进展,但还一贯不建立
WebSocket.OPEN :连接已经建立,可以发送新闻
WebSocket.CLOSING :连接正在进展关闭握手
WebSocket.CLOSED :连接已经关门或无法打开

除外在open事件回调中调用send方法,可因而判断readyState值来发送消息。

JavaScript

function bindEventHandler(data) { if (ws.readyState === WebSocket.OPEN)
{ ws.send(data); } else { //do something } }

1
2
3
4
5
6
7
function bindEventHandler(data) {
    if (ws.readyState === WebSocket.OPEN) {
        ws.send(data);
    } else {
        //do something
    }
}
JavaScript

ws.onopen = function () { setInterval( function() {
//缓存未满的时候发送 if (ws.bufferedAmount &lt; 1024 \* 5) {
ws.send(data); } }, 2000); };

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5b8f447934b7a325701025-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b7a325701025-2">
2
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b7a325701025-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b7a325701025-4">
4
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b7a325701025-5">
5
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b7a325701025-6">
6
</div>
<div class="crayon-num" data-line="crayon-5b8f447934b7a325701025-7">
7
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5b8f447934b7a325701025-8">
8
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5b8f447934b7a325701025-1" class="crayon-line">
ws.onopen = function () {
</div>
<div id="crayon-5b8f447934b7a325701025-2" class="crayon-line crayon-striped-line">
    setInterval( function() {
</div>
<div id="crayon-5b8f447934b7a325701025-3" class="crayon-line">
        //缓存未满的时候发送
</div>
<div id="crayon-5b8f447934b7a325701025-4" class="crayon-line crayon-striped-line">
        if (ws.bufferedAmount &lt; 1024 * 5) {
</div>
<div id="crayon-5b8f447934b7a325701025-5" class="crayon-line">
            ws.send(data);
</div>
<div id="crayon-5b8f447934b7a325701025-6" class="crayon-line crayon-striped-line">
        }
</div>
<div id="crayon-5b8f447934b7a325701025-7" class="crayon-line">
    }, 2000);
</div>
<div id="crayon-5b8f447934b7a325701025-8" class="crayon-line crayon-striped-line">
};
</div>
</div></td>
</tr>
</tbody>
</table>

接下去,大家说说握手阶段进程。

当大家成立Websocket实例对象与服务器建立连接时,

JavaScript

const ws = new WebSocket(‘ws://localhost:8023’);

1
const ws = new WebSocket(‘ws://localhost:8023’);

先是客户端向服务器发起3个握手请求,其请求报文的始末如下:

JavaScript

GET /game HTTP/1.1 Host: 10.242.17.102:8023 Cache-Control: no-cache
Upgrade: websocket Connection: Upgrade Sec-WebSocket-Key:
dGhlIHNhbXBsZSBub25jZQ== Sec-WebSocket-Protocol: game
Sec-WebSocket-Version: 10 Origin: http://192.168.185.16 Accept-Encoding:
gzip, deflate, sdch Accept-Language: zh-CN,zh;q=0.8

1
2
3
4
5
6
7
8
9
10
11
GET /game HTTP/1.1
Host: 10.242.17.102:8023
Cache-Control: no-cache
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Key: dGhlIHNhbXBsZSBub25jZQ==
Sec-WebSocket-Protocol: game
Sec-WebSocket-Version: 10
Origin: http://192.168.185.16
Accept-Encoding: gzip, deflate, sdch
Accept-Language: zh-CN,zh;q=0.8

从请求头中可以见见,其实是壹个基于http的握手请求。与常见的http请求例外的是,扩张了部分头消息。

服务器响应:

当服务器重回以下内容,就代表已经接受客户端请求啦,可以创制Websocket通讯啦。

JavaScript

HTTP/1.1 101 Switching Protocols Upgrade: websocket Connection: Upgrade
Sec-WebSocket-Accept: SIEylb7zRYJAEgiqJXaOW3V+ZWQ=

1
2
3
4
HTTP/1.1 101 Switching Protocols
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Accept: SIEylb7zRYJAEgiqJXaOW3V+ZWQ=

图片 4

迄今甘休,客户端和服务器握手成功建立了Websocket连接,通讯不再采用http数据帧,而利用Websocket独立的数据帧。


以上是Websocket商谈的基础理论篇I, 欢迎小伙伴儿们陆续(理论篇II,
实战篇神马的), 一起学学共同积累


1 赞 4 收藏
评论

图片 5

Websocket是什么:

MuranoFC中写到:WebSocket协议使在支配环境下运维不受信任代码的客户端和力所能及选用与那么些代码通讯的远程主机之间可以双向通信。

对,划重点:双向通讯

Websocket在连年之后,客户端可以主动发送新闻给服务器,服务器也足以当仁不让向客户端推送音讯。比如:预定车票音讯,除了大家发请求询问车票如何,当然更希望假设有新新闻,可以平昔公告大家。

其特点:

(1)握手阶段采纳 HTTP 协议,默认端口是80和443

(2)建立在TCP协议基础之上,和http协议同属于应用层

(4)可以发送文书,也可以发送二进制数据

(5)没有同源限制,客户端可以与自由服务器通讯

(6)协议标识符是ws(假设加密,为wss),如ws://localhost:8023

简短来说,Websocket商谈分成两有的:握手和多少传输。

图片 6

Websocket API:

此处是指客户端 API。

WebSocket 构造函数

透过调用WebSocket构造函数来创建三个WebSocket实例对象,建立客户端与服务器的接二连三。

JavaScript

const ws = new WebSocket(‘ws://localhost:8023’);

1
const ws = new WebSocket(‘ws://localhost:8023’);

Websocket事件

WebSocket 是纯事件驱动,通过监听事件可以拍卖到来的数据和改变的总是景况。服务端发送数据后,音讯和事件会异步到达。

JavaScript

// 连接请求open事件处理: ws.onopen = e =&gt; {
console.log('Connection success'); ws.send(\`Hello ${e}\`); };

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5a72977999e26474564720-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e26474564720-2">
2
</div>
<div class="crayon-num" data-line="crayon-5a72977999e26474564720-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e26474564720-4">
4
</div>
<div class="crayon-num" data-line="crayon-5a72977999e26474564720-5">
5
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5a72977999e26474564720-1" class="crayon-line">
 // 连接请求open事件处理:
</div>
<div id="crayon-5a72977999e26474564720-2" class="crayon-line crayon-striped-line">
     ws.onopen = e =&gt; {
</div>
<div id="crayon-5a72977999e26474564720-3" class="crayon-line">
         console.log('Connection success');
</div>
<div id="crayon-5a72977999e26474564720-4" class="crayon-line crayon-striped-line">
         ws.send(`Hello ${e}`);
</div>
<div id="crayon-5a72977999e26474564720-5" class="crayon-line">
     };
</div>
</div></td>
</tr>
</tbody>
</table>

如若要内定多个回调函数,可以行使add伊夫ntListener方法。

JavaScript

ws.addEventListener(‘open’, e => { ws.send(`Hello ${e}`); });

1
2
3
ws.addEventListener(‘open’, e => {
  ws.send(`Hello ${e}`);
});

当open事件触发时,意味着握手阶段已终止。服务端已经处理了连年的央求,可以准备收发数据。

JavaScript

// 接受文本消息的事件处理: ws.onmessage = e =&gt; { const data =
e.data; if (typeof data === "string") { console.log("Received string
message ",data); } else if (data instanceof Blob) {
console.log("Received blob message ", data); } };

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5a72977999e30543698552-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e30543698552-2">
2
</div>
<div class="crayon-num" data-line="crayon-5a72977999e30543698552-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e30543698552-4">
4
</div>
<div class="crayon-num" data-line="crayon-5a72977999e30543698552-5">
5
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e30543698552-6">
6
</div>
<div class="crayon-num" data-line="crayon-5a72977999e30543698552-7">
7
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e30543698552-8">
8
</div>
<div class="crayon-num" data-line="crayon-5a72977999e30543698552-9">
9
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5a72977999e30543698552-1" class="crayon-line">
// 接受文本消息的事件处理:
</div>
<div id="crayon-5a72977999e30543698552-2" class="crayon-line crayon-striped-line">
ws.onmessage = e =&gt; {
</div>
<div id="crayon-5a72977999e30543698552-3" class="crayon-line">
    const data = e.data;
</div>
<div id="crayon-5a72977999e30543698552-4" class="crayon-line crayon-striped-line">
    if (typeof data === &quot;string&quot;) {
</div>
<div id="crayon-5a72977999e30543698552-5" class="crayon-line">
        console.log(&quot;Received string message &quot;,data);
</div>
<div id="crayon-5a72977999e30543698552-6" class="crayon-line crayon-striped-line">
    } else if (data instanceof Blob) {
</div>
<div id="crayon-5a72977999e30543698552-7" class="crayon-line">
        console.log(&quot;Received blob message &quot;, data);
</div>
<div id="crayon-5a72977999e30543698552-8" class="crayon-line crayon-striped-line">
    }
</div>
<div id="crayon-5a72977999e30543698552-9" class="crayon-line">
};
</div>
</div></td>
</tr>
</tbody>
</table>

服务器数据只怕是文本,也恐怕是二进制数据,有Blob和ArrayBuffer两系列型,在读取到数量以前要求控制好数据的体系。

JavaScript

//异常处理 ws.onerror = e =&gt; { console.log("WebSocket Error: " ,
e); handleErrors(e); };

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5a72977999e34725915317-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e34725915317-2">
2
</div>
<div class="crayon-num" data-line="crayon-5a72977999e34725915317-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e34725915317-4">
4
</div>
<div class="crayon-num" data-line="crayon-5a72977999e34725915317-5">
5
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5a72977999e34725915317-1" class="crayon-line">
//异常处理
</div>
<div id="crayon-5a72977999e34725915317-2" class="crayon-line crayon-striped-line">
ws.onerror = e =&gt; {
</div>
<div id="crayon-5a72977999e34725915317-3" class="crayon-line">
    console.log(&quot;WebSocket Error: &quot; , e);
</div>
<div id="crayon-5a72977999e34725915317-4" class="crayon-line crayon-striped-line">
    handleErrors(e);
</div>
<div id="crayon-5a72977999e34725915317-5" class="crayon-line">
};
</div>
</div></td>
</tr>
</tbody>
</table>

WebSocket 规范中定义了ping 帧 和pong
帧,能够用来做心跳重连,互联网状态查询等,可是当前
浏览器只会自动发送pong帧,而不会发ping 帧。(有趣味可详查ping和pong帧)

JavaScript

//关闭连接处理 ws.onclose = e => { const code = e.code; const reason
= e.reason; console.log(“Connection close”, code, reason); };

1
2
3
4
5
6
//关闭连接处理
ws.onclose = e => {
    const code = e.code;
    const reason = e.reason;
    console.log("Connection close", code, reason);
};

WebSocket 方法:

WebSocket 对象有多个艺术:send 和 close

JavaScript

//发送一个文本消息 ws.send("this is websocket");

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5a72977999e3b847111158-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e3b847111158-2">
2
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5a72977999e3b847111158-1" class="crayon-line">
//发送一个文本消息
</div>
<div id="crayon-5a72977999e3b847111158-2" class="crayon-line crayon-striped-line">
ws.send(&quot;this is websocket&quot;);
</div>
</div></td>
</tr>
</tbody>
</table>

在open事件的回调中调用send()方法传送数据:

JavaScript

const ws = new WebSocket(‘ws://localhost:8023’); ws.onopen = e => {
console.log(‘Connection success’); ws.send(`Hello ${e}`); };

1
2
3
4
5
const ws = new WebSocket(‘ws://localhost:8023’);
ws.onopen = e => {
    console.log(‘Connection success’);
    ws.send(`Hello ${e}`);
};

假若想经过响应其他事件发送新闻,可透过判断当前的Websocket的readyState属性。接下来会说到readyState.

JavaScript

ws.close(); //1000是状态码,代表正常结束。 ws.close(1000, "Closing
normally");

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5a72977999e42958089470-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e42958089470-2">
2
</div>
<div class="crayon-num" data-line="crayon-5a72977999e42958089470-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e42958089470-4">
4
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5a72977999e42958089470-1" class="crayon-line">
ws.close();
</div>
<div id="crayon-5a72977999e42958089470-2" class="crayon-line crayon-striped-line">
 
</div>
<div id="crayon-5a72977999e42958089470-3" class="crayon-line">
//1000是状态码,代表正常结束。
</div>
<div id="crayon-5a72977999e42958089470-4" class="crayon-line crayon-striped-line">
ws.close(1000, &quot;Closing normally&quot;);
</div>
</div></td>
</tr>
</tbody>
</table>

WebSocket 属性

readyState值表示连接景况,是只读属性。它有以下五个值:

WebSocket.CONNECTING :连接正在进展,但还并未树立
WebSocket.OPEN :连接已经成立,可以发送音讯
WebSocket.CLOSING :连接正在开展关闭握手
WebSocket.CLOSED :连接已经关闭或无法打开

除却在open事件回调中调用send方法,可经过判断readyState值来发送消息。

JavaScript

function bindEventHandler(data) { if (ws.readyState === WebSocket.OPEN)
{ ws.send(data); } else { //do something } }

1
2
3
4
5
6
7
function bindEventHandler(data) {
    if (ws.readyState === WebSocket.OPEN) {
        ws.send(data);
    } else {
        //do something
    }
}
JavaScript

ws.onopen = function () { setInterval( function() {
//缓存未满的时候发送 if (ws.bufferedAmount &lt; 1024 \* 5) {
ws.send(data); } }, 2000); };

<table>
<colgroup>
<col style="width: 50%" />
<col style="width: 50%" />
</colgroup>
<tbody>
<tr class="odd">
<td><div class="crayon-nums-content" style="font-size: 13px !important; line-height: 15px !important;">
<div class="crayon-num" data-line="crayon-5a72977999e48105878083-1">
1
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e48105878083-2">
2
</div>
<div class="crayon-num" data-line="crayon-5a72977999e48105878083-3">
3
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e48105878083-4">
4
</div>
<div class="crayon-num" data-line="crayon-5a72977999e48105878083-5">
5
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e48105878083-6">
6
</div>
<div class="crayon-num" data-line="crayon-5a72977999e48105878083-7">
7
</div>
<div class="crayon-num crayon-striped-num" data-line="crayon-5a72977999e48105878083-8">
8
</div>
</div></td>
<td><div class="crayon-pre" style="font-size: 13px !important; line-height: 15px !important; -moz-tab-size:4; -o-tab-size:4; -webkit-tab-size:4; tab-size:4;">
<div id="crayon-5a72977999e48105878083-1" class="crayon-line">
ws.onopen = function () {
</div>
<div id="crayon-5a72977999e48105878083-2" class="crayon-line crayon-striped-line">
    setInterval( function() {
</div>
<div id="crayon-5a72977999e48105878083-3" class="crayon-line">
        //缓存未满的时候发送
</div>
<div id="crayon-5a72977999e48105878083-4" class="crayon-line crayon-striped-line">
        if (ws.bufferedAmount &lt; 1024 * 5) {
</div>
<div id="crayon-5a72977999e48105878083-5" class="crayon-line">
            ws.send(data);
</div>
<div id="crayon-5a72977999e48105878083-6" class="crayon-line crayon-striped-line">
        }
</div>
<div id="crayon-5a72977999e48105878083-7" class="crayon-line">
    }, 2000);
</div>
<div id="crayon-5a72977999e48105878083-8" class="crayon-line crayon-striped-line">
};
</div>
</div></td>
</tr>
</tbody>
</table>

接下去,我们说说握手阶段进程。

当大家创制Websocket实例对象与服务器建立连接时,

JavaScript

const ws = new WebSocket(‘ws://localhost:8023’);

1
const ws = new WebSocket(‘ws://localhost:8023’);

率先客户端向服务器发起一个握手请求,其请求报文的内容如下:

JavaScript

GET /game HTTP/1.1 Host: 10.242.17.102:8023 Cache-Control: no-cache
Upgrade: websocket Connection: Upgrade Sec-WebSocket-Key:
dGhlIHNhbXBsZSBub25jZQ== Sec-WebSocket-Protocol: game
Sec-WebSocket-Version: 10 Origin: http://192.168.185.16 Accept-Encoding:
gzip, deflate, sdch Accept-Language: zh-CN,zh;q=0.8

1
2
3
4
5
6
7
8
9
10
11
GET /game HTTP/1.1
Host: 10.242.17.102:8023
Cache-Control: no-cache
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Key: dGhlIHNhbXBsZSBub25jZQ==
Sec-WebSocket-Protocol: game
Sec-WebSocket-Version: 10
Origin: http://192.168.185.16
Accept-Encoding: gzip, deflate, sdch
Accept-Language: zh-CN,zh;q=0.8

从请求头中得以看看,其实是三个依据http的抓手请求。与一般的http请求例外的是,扩大了有的头音信。

服务器响应:

当服务器再次来到以下内容,就象征早已接受客户端请求啦,能够建立Websocket通讯啦。

JavaScript

HTTP/1.1 101 Switching Protocols Upgrade: websocket Connection: Upgrade
Sec-WebSocket-Accept: SIEylb7zRYJAEgiqJXaOW3V+ZWQ=

1
2
3
4
HTTP/1.1 101 Switching Protocols
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Accept: SIEylb7zRYJAEgiqJXaOW3V+ZWQ=

图片 7

从那之后,客户端和服务器握手成功建立了Websocket连接,通讯不再行使http数据帧,而利用Websocket独立的数据帧。


以上是Websocket合计的基础理论篇I, 欢迎小伙伴儿们陆续(理论篇II,
实战篇神马的), 一起学学共同积累


1 赞 3 收藏
评论

相关文章

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图