热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

开发笔记:5000字!带你零距离接触websocket!

本文由编程笔记#小编为大家整理,主要介绍了5000字!带你零距离接触websocket!相关的知识,希望对你有一定的参考价值。
本文由编程笔记#小编为大家整理,主要介绍了5000字!带你零距离接触websocket!相关的知识,希望对你有一定的参考价值。




作者@johnYu


原文 https://juejin.im/post/6876301731966713869


排版 执行上下文









1、什么是WebSocket?


1.1、定义



Websocket是一个持久化的网络通信协议,可以在单个 TCP 连接上进行全双工通讯,没有了Request和Response的概念,两者地位完全平等,连接一旦建立,客户端和服务端之间实时可以进行双向数据传输。



1.2、关联和区别


1.2.1、HTTP


1、HTTP是非持久的协议,客户端想知道服务端的处理进度只能通过不停地使用 Ajax进行轮询或者采用 long poll 的方式来,但是前者对服务器压力大,后者则会因为一直等待Response造成阻塞。




2、虽然http1.1默认开启了keep-alive长连接保持了这个TCP通道使得在一个HTTP连接中,可以发送多个Request,接收多个Response,但是一个request只能有一个response。而且这个response也是被动的,不能主动发起。


3、websocket虽然是独立于HTTP的一种协议,但是websocket必须依赖 HTTP 协议进行一次握手(在握手阶段是一样的),握手成功后,数据就直接从 TCP通道传输,与 HTTP 无关了,可以用一张图理解两者有交集,但是并不是全部。



5000字!带你零距离接触websocket!

1.2.2、socket


1、socket也被称为套接字,与HTTP和WebSocket不一样,socket不是协议,它是在程序层面上对传输层协议(可以主要理解为TCP/IP)的接口封装。可以理解为一个能够提供端对端的通信的调用接口(API)


1.3、应用场景



WebSocket可以做弹幕、消息订阅、多玩家游戏、协同编辑、股票基金实时报价、视频会议、在线教育、聊天室等应用实时监听服务端变化。



1.3.1、Websocket握手


// 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 报文不同的地方:


Upgrade: websocket
Connection: Upgrade

表示发起的是 WebSocket 协议


Sec-WebSocket-Key: x3JJHMbDL1EzLkh9GBhXDw==
Sec-WebSocket-Protocol: chat, superchat
Sec-WebSocket-Version: 13

Sec-WebSocket-Key 是由浏览器随机生成的,验证是否可以进行Websocket通信,防止恶意或者无意的连接。


Sec_WebSocket-Protocol 是用户自定义的字符串,用来标识服务所需要的协议


Sec-WebSocket-Version 表示支持的 WebSocket 版本。


// 服务器响应:☟ ☟ ☟
HTTP/1.1 101 Switching Protocols
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Accept: HSmrc0sMlYUkAGmm5OPpG2HaGWk=
Sec-WebSocket-Protocol: chat

101 响应码 表示要转换协议。


Connection: Upgrade 表示升级新协议请求。


Upgrade: websocket 表示升级为 WebSocket 协议。


Sec-WebSocket-Accept 是经过服务器确认,并且加密过后的


Sec-WebSocket-Key 用来证明客户端和服务器之间能进行通信了。


Sec-WebSocket-Protocol 表示最终使用的协议。



至此,客户端和服务器握手成功建立了Websocket连接,HTTP已经完成它所有工作了,接下来就是完全按照Websocket协议进行通信了。



2、关于Websocket


2.1、WebSocket心跳


可能会有一些未知情况导致SOCKET断开,而客户端和服务端却不知道,需要客户端定时发送一个心跳 Ping 让服务端知道自己在线,而服务端也要回复一个心跳 Pong告诉客户端自己可用,否则视为断开。


2.2、WebSocket状态


WebSocket 对象中的readyState属性有四种状态:
0: 表示正在连接
1: 表示连接成功,可以通信了
2: 表示连接正在关闭
3: 表示连接已经关闭,或者打开连接失败

2.3、WebSocket实践


2.3.1、服务端接收发送消息


WebSocket的服务端部分,本文会以Node.js搭建。


安装express和负责处理WebSocket协议的ws:


npm install express ws

安装成功后的package.json:


5000字!带你零距离接触websocket!

接着在根目录创建server.js文件:

// 引入express 和 ws
const express = require('express');
const SocketServer = require('ws').Server;
// 指定开启的端口号const PORT = 3000;
// 创建express,绑定监听3000端口,且设定开启后在consol中提示
const server = express().listen(PORT, () => console.log(`Listening on ${PORT}`));
// 将express交给SocketServer开启WebSocket的服务
const wss = new SocketServer({ server });
// 当 WebSocket 从外部连接时执行
wss.on('connection', (ws) => {
    // 连接时执行此 console 提示 console.log('Client connected');
    // 对message设置监听,接收从客户端发送的消息 
    ws.on('message', (data) => {
    
        // data为客户端发送的消息,将消息原封不动返回回去
        ws.send(data);
    });
    
    // 当WebSocket的连接关闭时执行
    ws.on('close', () => {
        console.log('Closeconnected');
    });
});

执行node server.js启动服务,端口打开后会执行监听时间打印提示,说明服务启动成功。


5000字!带你零距离接触websocket!

在开启WebSocket后,服务端会在message中监听,接收参数data捕获客户端发送的消息,然后使用send发送消息。

2.3.2、客户端接收发送消息


分别在根目录创建index.html和index.js文件

// index.html

  
    
  


// index.js
// 使用WebSocket的地址向服务端开启连接
let ws = new WebSocket('ws://localhost:3000');
// 开启后的动作,指定在连接后执行的事件
ws.onopen = () => {
    console.log('open connection');
};
// 接收服务端发送的消息
ws.onmessage = (event) => {
    console.log(event);
};
// 指定在关闭后执行的事件
ws.onclose = () => {
    console.log('close connection');
};


5000字!带你零距离接触websocket!

打印了open connection说明连接成功,客户端会使用onmessage处理接收。

其中event参数包含这次沟通的详细信息,从服务端回传的消息会在event的data属性中。

手动在控制台调用send发送消息,打印event回传信息:


5000字!带你零距离接触websocket!


2.3.3、服务端定时发送


上面是从客户端发送消息,服务端回传。我们也可以通过setInterval让服务端在固定时间发送消息给客户端:

// server.js修改如下:
// 当WebSocket从外部连接时执行
wss.on('connection', (ws) => {
    // 连接时执行此 console 提示
    console.log('Client connected');
    
    // 固定发送最新消息给客户端
    const = sendNowTime = setInterval(() => {
        ws.send(String(new Date()));
    }, 1000);
    
    // 对message设置监听,接收从客户端发送的消息
    ws.on('message', (data) => {
    
        // data为客户端发送的消息,将消息原封不动返回回去
        ws.send(data);
    });
    
    // 当WebSocket的连接关闭时执行
    ws.on('close',() => {
        console.log('Close connected');
    });
});

客户端连接后就会定时接收,直至我们关闭websocket服务。


5000字!带你零距离接触websocket!

2.3.4、多人聊天


如果多个客户端连接按照上面的方式只会返回各自发送的消息,先注释服务端定时发送,开启两个窗口模拟:


5000字!带你零距离接触websocket!

如果我们要让客户端间消息共享,也同时接收到服务端回传的消息呢?


我们可以使用clients找出当前所有连接中的客户端 ,并通过回传消息发送到每一个客户端中:


修改server.js如下:

//当WebSocket从外部连接时执行
wss.on('connection', (ws) => {
    //连接时执行此 console 提示
    console.log('Client connected');
    
    //固定发送最新消息给客户端
    const sendNowTime = setInterval(() => {
        ws.send(String(new Date()));
    }, 1000);
    //对message设置监听,接收从客户端发送的消息
    ws.on('message', (data) => {
    
        //取得所有连接中的 客户端
        let clients = wss.clients;
        
        //循环,发送消息至每个客户端
        clients.forEach((client) => {
            client.send(data);
        });
    });
    //当WebSocket的连接关闭时执行
    ws.on('close', () => {
        console.log('Close connected');
    });
});

这样一来,不论在哪个客户端发送消息,服务端都能将消息回传到每个客户端:


5000字!带你零距离接触websocket!

可以观察下连接信息:


5000字!带你零距离接触websocket!

3、总结

推荐阅读
  • 本文介绍了在rhel5.5操作系统下搭建网关+LAMP+postfix+dhcp的步骤和配置方法。通过配置dhcp自动分配ip、实现外网访问公司网站、内网收发邮件、内网上网以及SNAT转换等功能。详细介绍了安装dhcp和配置相关文件的步骤,并提供了相关的命令和配置示例。 ... [详细]
  • Nginx使用AWStats日志分析的步骤及注意事项
    本文介绍了在Centos7操作系统上使用Nginx和AWStats进行日志分析的步骤和注意事项。通过AWStats可以统计网站的访问量、IP地址、操作系统、浏览器等信息,并提供精确到每月、每日、每小时的数据。在部署AWStats之前需要确认服务器上已经安装了Perl环境,并进行DNS解析。 ... [详细]
  • Linux服务器密码过期策略、登录次数限制、私钥登录等配置方法
    本文介绍了在Linux服务器上进行密码过期策略、登录次数限制、私钥登录等配置的方法。通过修改配置文件中的参数,可以设置密码的有效期、最小间隔时间、最小长度,并在密码过期前进行提示。同时还介绍了如何进行公钥登录和修改默认账户用户名的操作。详细步骤和注意事项可参考本文内容。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • 图解redis的持久化存储机制RDB和AOF的原理和优缺点
    本文通过图解的方式介绍了redis的持久化存储机制RDB和AOF的原理和优缺点。RDB是将redis内存中的数据保存为快照文件,恢复速度较快但不支持拉链式快照。AOF是将操作日志保存到磁盘,实时存储数据但恢复速度较慢。文章详细分析了两种机制的优缺点,帮助读者更好地理解redis的持久化存储策略。 ... [详细]
  • 本文介绍了RPC框架Thrift的安装环境变量配置与第一个实例,讲解了RPC的概念以及如何解决跨语言、c++客户端、web服务端、远程调用等需求。Thrift开发方便上手快,性能和稳定性也不错,适合初学者学习和使用。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了如何使用iptables添加非对称的NAT规则段,以实现内网穿透和端口转发的功能。通过查阅相关文章,得出了解决方案,即当匹配的端口在映射端口的区间内时,可以成功进行端口转发。详细的操作步骤和命令示例也在文章中给出。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • 本文介绍了计算机网络的定义和通信流程,包括客户端编译文件、二进制转换、三层路由设备等。同时,还介绍了计算机网络中常用的关键词,如MAC地址和IP地址。 ... [详细]
  • Python瓦片图下载、合并、绘图、标记的代码示例
    本文提供了Python瓦片图下载、合并、绘图、标记的代码示例,包括下载代码、多线程下载、图像处理等功能。通过参考geoserver,使用PIL、cv2、numpy、gdal、osr等库实现了瓦片图的下载、合并、绘图和标记功能。代码示例详细介绍了各个功能的实现方法,供读者参考使用。 ... [详细]
  • 利用Visual Basic开发SAP接口程序初探的方法与原理
    本文介绍了利用Visual Basic开发SAP接口程序的方法与原理,以及SAP R/3系统的特点和二次开发平台ABAP的使用。通过程序接口自动读取SAP R/3的数据表或视图,在外部进行处理和利用水晶报表等工具生成符合中国人习惯的报表样式。具体介绍了RFC调用的原理和模型,并强调本文主要不讨论SAP R/3函数的开发,而是针对使用SAP的公司的非ABAP开发人员提供了初步的接口程序开发指导。 ... [详细]
  • 本文介绍了在mac环境下使用nginx配置nodejs代理服务器的步骤,包括安装nginx、创建目录和文件、配置代理的域名和日志记录等。 ... [详细]
  • WebSocket与Socket.io的理解
    WebSocketprotocol是HTML5一种新的协议。它的最大特点就是,服务器可以主动向客户端推送信息,客户端也可以主动向服务器发送信息,是真正的双向平等对话,属于服务器推送 ... [详细]
author-avatar
贤闲咸大_552
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有