1、什么是WebSocket

WebSocket 是一种自然的全双工、双向、单套接字连接。使用WebSocket,你的HTTP 请求变成打开WebSocket 连接(WebSocket 或者WebSocket over TLS(TransportLayer Security,传输层安全性,原称“SSL”))的单一请求,并且重用从客户端到服务器以及服务器到客户端的同一连接。WebSocket 减少了延迟,因为一旦建立起WebSocket 连接,服务器可以在消息可用时发送它们。例如,和轮询不同,WebSocket只发出一个请求。服务器不需要等待来自客户端的请求。相似地,客户端可以在任何时候向服务器发送消息。相比轮询不管是否有可用消息,每隔一段时间都发送一个请求,单一请求大大减少了延迟。

2、WebSocket API

WebSocket API 使你可以通过Web,在客户端应用程序和服务器端进程之间建立全双工的双向通信。WebSocket 接口规定了可用于客户端的方法以及客户端与网络的交互方式。首先,你要调用WebSocket 构造函数(constructor),创建一个WebSocket 连接。构造函数返回WebSocket 对象实例。你可以监听该对象上的事件,这些事件告诉你何时连接打开,何时消息到达,何时连接关闭以及何时发生错误。你可以与WebSocket 对象交互,发送消息或者关闭连接。下面来研究WebSocket API 的各个方面。

3、WebSocket 构造函数

为了建立到服务器的WebSocket 连接,使用WebSocket 接口,通过指向一个代表所要连接端点的URL,实例化一个WebSocket对象。WebSocket 协议定义了两种URL 方案(URL scheme)—ws 和wss,分别用于客户端和服务器之间的非加密与加密流量。ws(WebSocket) 方案与HTTP URI 方案类似。wss(WebSocketSecure,WebSocket 安全)URI 方案表示使用传输层安全性(TLS,也叫SSL)的WebSocket 连接,使用HTTPS 采用的安全机制来保证HTTP 连接的安全。WebSocket 构造函数有一个必需的参数URL(指向连接目标的URL)和一个可选参数protocols(为了建立连接,服务器必须在其响应中包含的一个或一组协议名称)。在protocols 参数中可以使用的协议包括XMPP(eXtensible Messaging and PresenceProtocol, 可扩展消息处理现场协议)、SOAP(Simple ObjectAccess Protocol,简单对象访问协议)或者自定义协议。

  1. var ws = new WebSocket("ws://www.websocket.org");

带有协议支持的WebSocket 构造函数示例

  1. // Connecting to the server with multiple protocol choices
  2. var echoSocket = new WebSocket("ws://echo.websocket.org", ["com.kaazing.echo",
  3. "example.imaginary.protocol"])
  4. echoSocket.onopen = function(e) {
  5. // Check the protocol chosen by the server
  6. console.log(echoSocket.protocol);
  7. }

由于WebSocket 服务器ws://echo.websocket.org 只理解com.kaazing.echo 协议, 而不理解example.imaginary.protocol,该服务器在触发WebSocket open 事件的时候选择com.kaazing.echo 协议。使用数组为你提供了让应用程序对不同服务器使用不同协议的灵活性。

4、WebSocket 事件

WebSocket API 是纯事件驱动的。应用程序代码监听WebSocket对象上的事件,以便处理输入数据和连接状态的改变。WebSocket协议也是事件驱动的。客户端应用程序不需要轮询服务器来得到更新的数据。消息和事件将在服务器发送它们的时候异步到达。WebSocket 编程遵循异步编程模式,也就是说,只要WebSocket连接打开,应用程序就简单地监听事件。客户端不需要主动轮询服务器得到更多的信息。要开始监听事件,只要为WebSocket 对象添加回调函数。也可以使用addEventListener() DOM 方法为WebSocket 对象添加事件监听器。

WebSocket 对象调度4 个不同的事件:
open
message
error
close
和所有Web API 一样,可以用on< 事件名称> 处理程序属性

监听这些事件,也可以使用addEventListener(); 方法。

4.1、WebSocket 事件:open

一旦服务器响应了WebSocket 连接请求,open 事件触发并建立一个连接。open 事件对应的回调函数称作onopen。

  1. ws.onopen = function(e) {
  2. console.log("Connection open...");
  3. };

到open 事件触发时,协议握手已经完成,WebSocket 已经准备好发送和接收数据。如果应用程序接收到一个open 事件,那么可以确定WebSocket 服务器成功地处理了连接请求,并且同意与应用程序通信。

4.2、WebSocket messagess事件

WebSocket 消息包含来自服务器的数据。你也可能听说过组成WebSocket 消息的WebSocket 帧(Frame)。message 事件在接收到消息时触发,对应于该事件的回调函数是onmessage。

  1. ws.onmessage = function(e) {
  2. if(typeof e.data === "string"){
  3. console.log("String message received", e, e.data);
  4. } else {
  5. console.log("Other message received", e, e.data);
  6. }
  7. };

除了文本,WebSocket 消息还可以处理二进制数据,这种数据作为Blob 消息或者ArrayBuffer 消息处理。因为设置WebSocket 消息二进制数据类型的应用程序会影响二进制消息,所以必须在读取数据之前决定用于客户端二进制输入数据的类型。

  1. ws.binaryType = "blob";
  2. // Event handler for receiving Blob messages
  3. ws.onmessage = function(e) {
  4. if(e.data instanceof Blob){
  5. console.log("Blob message received", e.data);
  6. var blob = new Blob(e.data);
  7. }
  8. ws.binaryType = "arraybuffer";
  9. // Event handler for receiving ArrayBuffer messages
  10. ws.onmessage = function(e) {
  11. if(e.data instanceof ArrayBuffer){
  12. console.log("ArrayBuffer Message Received", + e.data);
  13. // e.data is an ArrayBuffer. Create a byte view of that object.
  14. var a = new Uint8Array(e.data);
  15. }
  16. };

4.3、 WebSocket 事件:error

error 事件在响应意外故障的时候触发。与该事件对应的回调函数为onerror。错误还会导致WebSocket 连接关闭。如果你接收一个error 事件,可以预期很快就会触发close 事件。close 事件中的代码和原因有时候能告诉你错误的根源。error事件处理程序是调用服务器重连逻辑以及处理来自WebSocket 对象的异常的最佳场所。

  1. ws.onerror = function(e){
  2. console.log('websocked error');
  3. handerError();
  4. }

4.4、 WebSocket 事件:close

close 事件在WebSocket 连接关闭时触发。对应于close 事件的回调函数是onclose。一旦连接关闭,客户端和服务器不再能接收或者发送消息。

  1. ws.onclose = function(e) {
  2. console.log("Connection closed", e);
  3. };

WebSocket close 事件在连接关闭时触发,这可能有多种原因,比如连接失败或者成功的WebSocket 关闭握手。WebSocket 对象特性readyState 反映了连接的状态(2 为正在关闭,3 为已关闭)。

close 事件有3 个有用的属性(property),可以用于错误处理和恢复:wasClean、code 和error。wasClean 属性是一个布尔属性,表示连接是否顺利关闭。如果WebSocket 的关闭是对来自服务器的一个close 帧的响应,则该属性为true。如果连接是因为其他原因(例如,因为底层TCP 连接关闭)关闭,则该属性为false。code 和reason 属性表示服务器发送的关闭握手状态。这些属性和WebSocket.close() 方法中的code 和reason 参数一致。

5、WebSocket 方法

WebSocket 对象有两个方法:send() 和close()。

5.1、WebSocket 方法:send()

使用WebSocket 在客户端和服务器之间建立全双工双向连接后,就可以在连接打开时(也就是说,在调用onopen 监听器之后,调用onclose 监听器之前)调用send() 方法。使用send() 方法可以从客户端向服务器发送消息。在发送一条或者多条消息之后,可以保持连接打开,或者调用close() 方法终止连接。

  1. ws.send("Hello WebSocket!");

send() 方法在连接打开的时候发送数据。如果连接不可用或者关闭,它抛出一个有关无效连接状态的异常。人们开始使用WebSocket API 时常犯的一个错误是试图在连接打开之前发送消息。

  1. // Wait until the open event before calling send().
  2. var ws = new WebSocket("ws://echo.websocket.org")
  3. ws.onopen = function(e) {
  4. ws.send("Initial data");
  5. }

如果想发送消息响应另一个事件, 可以检查WebSocketreadyState 属性,并选择只在套接字打开时发送数据。

  1. function myEventHandler(data) {
  2. if (ws.readyState === WebSocket.OPEN) {
  3. // The socket is open, so it is ok to send the data.
  4. ws.send(data);
  5. } else {
  6. // Do something else in this case.
  7. //Possibly ignore the data or enqueue it.
  8. }
  9. }

除了文本(字符串)消息之外,WebSocket API 允许发送二进制数据,这对于实现二进制协议特别有用。这样的二进制协议可能是TCP 上层的标准互联网协议,这些协议的载荷可能是Blob 或ArrayBuffer。

5.2、WebSocket 方法:close()

使用close() 方法, 可以关闭WebSocket 连接或者终止连接尝试。如果连接已经关闭,该方法就什么都不做。在调用close() 之后,不能在已经关闭的WebSocket 上发送任何数据。可以向close() 方法传递两个可选参数:code(数字型的状态代码)和reason(一个文本字符串)。传递这些参数能够向服务器传递关于客户关闭连接原因的信息。

6、WebSocket 对象特性

可以使用多种WebSocket 对象特性提供关于WebSocket 对象的更多信息:readyState、bufferedAmount 和protocol。

6.1、WebSocket 对象特性:readyState

下表readyState 特性、取值和状态描述

特性常量 取值 状态
WebSocket.CONNECTING 0 连接正在进行中,但还未建立
WebSocket.OPEN 1 连接已经建立。消息可以在客户端和服务器之间传递
WebSocket.CLOSING 2 连接正在进行关闭握手
WebSocket.CLOSED 3 连接已经关闭,不能打开

6.2、WebSocket 对象特性:bufferedAmount

设计应用程序时,你可能想要检查发往服务器的缓冲数据量,特别是在客户端应用程序向服务器发送大量数据的时候。尽管调用send() 是立即生效的,但是数据在互联网上的传输却不是如此。浏览器将为你的客户端应用程序缓存出站数据,从而使你可以随时调用send(),发送任意数量的数据。然而,如果你想知道数据在网络上传送的速率,WebSocket 对象可以告诉你缓存的大小。你可以使用bufferedAmount 特性检查已经进入队列,但是尚未发送到服务器的字节数。这个特性报告的值不包括协议组帧开销或者操作系统、网络硬件所进行的缓冲。

代码展示一个使用bufferedAmount 特性每秒发送更新的例子。如果网络无法承受这一速率,它会相应地作出调整。

  1. // 10k max buffer size.
  2. var THRESHOLD = 10240;
  3. // Create a New WebSocket connection
  4. var ws = new WebSocket("ws://echo.websocket.org/updates");
  5. // Listen for the opening event
  6. ws.onopen = function () {
  7. // Attempt to send update every second.
  8. setInterval( function() {
  9. // Send only if the buffer is not full
  10. if (ws.bufferedAmount < THRESHOLD) {
  11. ws.send(getApplicationState());
  12. }
  13. }, 1000);
  14. };

对于限制应用向服务器发送数据的速率,从而避免网络饱和,bufferedAmount 特性很有用。

6.3、 WebSocket 对象特性:protocol

在前面关于WebSocket 构造函数的讨论中, 我们提到了protocol 参数,它让服务器知道客户端理解并可在WebSocket上使用的协议。WebSocket 对象的protocol 特性提供了另一条关于WebSocket 实例的有用信息。客户端和服务器协议协商的结果可以在WebSocket 对象上看到。protocol 特性包含在打开握手期间WebSocket 服务器选择的协议名,换句话说,protocol特性告诉你特定WebSocket 上使用的协议。protocol 特性在最初的握手完成之前为空,如果服务器没有选择客户端提供的某个协议,该特性保持空值。

例子:

  1. <!DOCTYPE html>
  2. <title>WebSocket Echo Client</title>
  3. <h2>Websocket Echo Client</h2>
  4. <div id="output"></div>
  5. <script>
  6. // Initialize WebSocket connection and event handlers
  7. function setup() {
  8. output = document.getElementById("output");
  9. ws = new WebSocket("ws://echo.websocket.org/echo");
  10. // Listen for the connection open event then call the sendMessage function
  11. ws.onopen = function(e) {
  12. log("Connected");
  13. sendMessage("这是发送的数据")
  14. }
  15. // Listen for the close connection event
  16. ws.onclose = function(e) {
  17. log("Disconnected: " + e.reason);
  18. }
  19. // Listen for connection errors
  20. ws.onerror = function(e) {
  21. log("Error ");
  22. }
  23. // Listen for new messages arriving at the client
  24. ws.onmessage = function(e) {
  25. log("Message received: " + e.data);
  26. // Close the socket once one message has arrived.
  27. ws.close();
  28. }
  29. }
  30. // Send a message on the WebSocket.
  31. function sendMessage(msg){
  32. ws.send(msg);
  33. log("Message sent");
  34. }
  35. // Display logging information in the document.
  36. function log(s) {
  37. var p = document.createElement("p");
  38. p.style.wordWrap = "break-word";
  39. p.textContent = s;
  40. output.appendChild(p);
  41. // Also log information on the javascript console
  42. console.log(s);
  43. }
  44. // Start running the example.
  45. setup();
  46. </script>
  47. </html>

7、WebSocket 浏览器兼容性检测

  1. if (window.WebSocket){
  2. console.log("This browser supports WebSocket!");
  3. } else {
  4. console.log("This browser does not support WebSocket.");
  5. }

8、在WebSocket 中使用HTML5 媒体

作为HTML5 和Web 平台的一部分,WebSocket API 可以很好地和所有HTML5 特性(feature)配合。这个API 所能发送和接收的数据类型广泛地用于传输应用程序数据和媒体。字符串当然可以表示XML 和JSON 等Web 数据格式。二进制类型可以和拖放(Drag-and-Drop)、FileReader、WebGL 和Web Audio API 等集成。我们来看看如何结合WebSocket 使用HTML5 媒体。代码清单展示了一个结合WebSocket 使用HTML5 媒体的完整客户端应用程序。

你可以根据这些代码创建自己的HTML 文件。

  1. <!DOCTYPE html>
  2. <title>WebSocket Image Drop</title>
  3. <h1>Drop Image Here</h1>
  4. <script>
  5. // Initialize WebSocket connection
  6. var wsUrl = "ws://echo.websocket.org/echo";
  7. var ws = new WebSocket(wsUrl);
  8. ws.onopen = function() {
  9. console.log("open");
  10. }
  11. // Handle binary image data received on the WebSocket
  12. ws.onmessage = function(e) {
  13. var blob = e.data;
  14. console.log("message: " + blob.size + " bytes");
  15. // Work with prefixed URL API
  16. if (window.webkitURL) {
  17. URL = webkitURL;
  18. }
  19. var uri = URL.createObjectURL(blob);
  20. var img = document.createElement("img");
  21. img.src = uri;
  22. document.body.appendChild(img);
  23. }
  24. // Handle drop event
  25. document.ondrop = function(e) {
  26. document.body.style.backgroundColor = "#fff";
  27. try {
  28. e.preventDefault();
  29. handleFileDrop(e.dataTransfer.files[0]);
  30. return false;
  31. } catch(err) {
  32. console.log(err);
  33. }
  34. }
  35. // Provide visual feedback for the drop area
  36. document.ondragover = function(e) {
  37. e.preventDefault();
  38. document.body.style.backgroundColor = "#6fff41";
  39. }
  40. document.ondragleave = function() {
  41. document.body.style.backgroundColor = "#fff";
  42. }
  43. // Read binary file contents and send them over WebSocket
  44. function handleFileDrop(file) {
  45. var reader = new FileReader();
  46. reader.readAsArrayBuffer(file);
  47. reader.onload = function() {
  48. console.log("sending: " + file.name);
  49. ws.send(reader.result);
  50. }
  51. }
  52. </script>
  53. </html>

文章参考:HTML5+WebSocket权威指南

学习html5的WebSocket连接的更多相关文章

  1. HTML5学习总结-08 WebSocket 服务器推送

    一 WebSocket 随着互联网的发展,传统的HTTP协议已经很难满足Web应用日益复杂的需求了.近年来,随着HTML5的诞生,WebSocket协议被提出,它实现了浏览器与服务器的全双工通信,扩展 ...

  2. HTML5的Websocket(理论篇 I)

    HTML5的Websocket(理论篇 I) ** 先请来TA的邻居:** http:无状态.基于tcp请求/响应模式的应用层协议 (A:哎呀,上次你请我吃饭了么? B:我想想, 上次请你吃了么) t ...

  3. HTML5之WebSocket && https://zhuanlan.zhihu.com/p/23467317

    在认识websocket之前,我们必须了解的是websocket有什么用? 他能解决我们遇到的什么问题? 如果没用,那么我们就么有使用它的必要的. websocket就是建立起全双工协议的,提高了效率 ...

  4. HTML5之WebSocket(转自知乎)

    在认识websocket之前,我们必须了解的是websocket有什么用? 他能解决我们遇到的什么问题? 如果没用,那么我们就么有使用它的必要的. websocket就是建立起全双工协议的,提高了效率 ...

  5. 大熊君学习html5系列之------Online && Offline(在线状态检测)

    一,开篇分析 Hi,大家好,给大家拜个晚年!大熊君又和大家见面了,(*^__^*) 嘻嘻……,这系列文章主要是学习Html5相关的知识点,以学习API知识点为入口,由浅入深的引入实例,让大家一步一步的 ...

  6. JavaEE7 HTML5利用WebSocket实现即时通讯

    HTML5给Web浏览器带来了全双工TCP连接websocket标准服务器的能力. 换句话说,浏览器能够与服务器建立连接,通过已建立的通信信道来发送和接收数据而不需要由HTTP协议引入额外其他的开销来 ...

  7. 认识HTML5的WebSocket

    在HTML5规范中,我最喜欢的Web技术就是正迅速变得流行的WebSocket API.WebSocket提供了一个受欢迎的技术,以替代我们过去几年一直在用的Ajax技术.这个新的API提供了一个方法 ...

  8. HTML5之WebSocket

    在HTML5规范中,我最喜欢的Web技术就是正迅速变得流行的WebSocket API.WebSocket提供了一个受欢迎的技术,以替代我们过去几年一直在用的Ajax技术.这个新的API提供了一个方法 ...

  9. 怎样学习HTML5?

    怎样学习HTML5?这个话题,问的人非常多,随便百度一下就能看到各种各样的回答.只是感觉每种回答都不给力.以下我给出一个自己理解的HTML5学习的路线图,依照这个路线图学习以后,一般的HTML5项目开 ...

随机推荐

  1. Repeater在无数据记录时显示暂无数据

    原文:Repeater在无数据记录时显示暂无数据 方法就是在FooterTemplate加个Label并根据repeater.Items.Count判断是否有记录.关键代码如下: <Footer ...

  2. 谈话ZooKeeper(一个)分析ZooKeeper的Quorums机制--预防Split-Brain问题

    使用ZooKeeper学生们应该看到一个参数.它是ZooKeeper超过一半的群集必须节点(Majority)可用的.外来人才在整个集群中可用.在大多数情况下,这种说法是正确的. 谈论这篇文章背后的原 ...

  3. 自己的包poi操作Excel工具

    在前面的文章<使用poi读写Excel>中分享了一下poi操作Excel的简单演示样例.这次要分享一下我封装的一个Excel操作的工具类. 该工具类主要完毕的功能是:读取Excel.汇总E ...

  4. .net下几种常用的对称加解密

    加密解密算法,太常用了,可是实在忍受不了十次八次之后还是要重头写,总是能告诉我原来的算法不好用(实际是压根不会用)的情况,不如直接写出来,再有人要直接给他看看,也顺便记录下算法,方便下新手大众. DE ...

  5. Android采用Volley具体的例子展示完整的异步加载数据(一)

    MainActivity例如下列: package cc.cn; import java.util.HashMap; import org.json.JSONObject; import androi ...

  6. 【Socket规划】套接字Windows台C语言

    [编译环境]:Visual Studio 2013 这是服务端实现流程. #include<stdio.h> #include<stdlib.h> #include<wi ...

  7. ubunut 查看port被哪个程序占用

    查看8087port被哪个程序占用 lsof -i :8087 -n

  8. 第4章 建造者模式(Builder Pattern)

    原文 第4章 建造者模式(Builder Pattern) 定义 将一个复杂对象的构造与它的表示分离,使同样的构建过程可以创建不同的表示,这样的设计模式被称为建造者模式. 实用范围 1 当创建复杂对象 ...

  9. jquery 元素控制(附加元素/其他内容)引进和应用

    一个.在内部元素/外部附加元件 append,prepend:加入到该子元素  before,after:元素加入 html: <div id="content"> 在 ...

  10. sql server中的索引详情

    什么是索引 拿汉语字典的目录页(索引)打比方:正如汉语字典中的汉字按页存放一样,SQL Server中的数据记录也是按页存放的,每页容量一般为4K .为了加快查找的速度,汉语字(词)典一般都有按拼音. ...