五分钟学会HTML5的WebSocket协议
<h2 id="1背景">1、背景</h2><p> 很多网站为了实现推送技术,所用的技术都是<strong>Ajax轮询</strong>。轮询是在特定的的时间间隔由浏览器对服务器发出HTTP请求,然后由服务器返回最新的数据给客户端的浏览器。这种传统的模式带来很明显的缺点,即浏览器需要不断的向服务器发出请求,然而HTTP请求可能包含较长的头部,其中真正有效的数据可能只是很小的一部分,显然这样会浪费很多的带宽等资源。HTML5新增的一些新协议<strong>WebSocket</strong>,可以提供在单个TCP连接上<strong>提供全双工,双向通信</strong>,能够节省服务器资源和带宽,并且能够实时进行通信。</p>
<h2 id="2websocket介绍">2、WebSocket介绍</h2>
<p> 传统的http也是一种协议,WebSocket是一种协议,使用http服务器无法实现WebSocket,</p>
<p><strong>2.1.浏览器支持情况</strong></p>
<p>基本主流浏览器都支持</p>
<p><img src="https://img2018.cnblogs.com/blog/1103967/201911/1103967-20191122022640237-1033309185.png" alt="" loading="lazy"></p>
<p><strong>2.2.优点</strong></p>
<p>相对于http有如下好处:</p>
<ul>
<li>1.客户端与服务器只建立一个TCP连接,可以使用更少的连接。</li>
<li>2.WebSocket服务器端可以主动推送数据到客户端,更灵活高效。</li>
<li>3.更轻量级的协议头,减少数据传送量。</li>
</ul>
<p>对比轮训机制</p>
<p><img src="https://img2018.cnblogs.com/blog/1103967/201911/1103967-20191122022658213-1496251397.png" alt="" loading="lazy"></p>
<h2 id="3websocket用法">3、WebSocket用法</h2>
<p> 我们了解WebSocket是什么,有哪些优点后,怎么使用呢?</p>
<p><strong>3.1.WebSocket创建</strong></p>
<p>WebSocket使用了自定义协议,url模式与http略有不同,未加密的连接是ws://,加密的连接是wss://,WebSocket实例使用<code>new WebSocket()</code>方法来创建,</p>
<pre><code>var ws = new WebSocket(url, );
</code></pre>
<p><em>第一个参数 url, 指定连接的 URL。第二个参数 protocol 是可选的,指定了可接受的子协议。</em></p>
<p><strong>3.2.WebSocket属性</strong></p>
<p>当创建ws对象后,readyState为ws实例状态,共4种状态</p>
<ul>
<li>
<p>0 表示连接尚未建立。</p>
</li>
<li>
<p>1 表示连接已建立,可以进行通信。</p>
</li>
<li>
<p>2 表示连接正在进行关闭。</p>
</li>
<li>
<p>3 表示连接已经关闭或者连接不能打开。</p>
</li>
</ul>
<p><strong>Tips:</strong>在发送报文之前要判断状态,断开也应该有重连机制。</p>
<p><strong>3.3.WebSocket事件</strong></p>
<p>在创建ws实例对象后,会拥有以下几个事件,根据不同状态可在事件回调写方法。</p>
<ul>
<li>
<p>ws.onopen 连接建立时触发</p>
</li>
<li>
<p>ws.onmessage 客户端接收服务端数据时触发</p>
</li>
<li>
<p>ws.onerror 通信发生错误时触发</p>
</li>
<li>
<p>ws.onclose 连接关闭时触发</p>
</li>
</ul>
<pre><code>ws.onmessage = (res) => {
console.log(res.data);
};
ws.onopen = () => {
console.log('OPEN...');
};
ws.onclose=()=>{
console.log('CLOSE...');
}
</code></pre>
<p><strong>3.4.WebSocket方法</strong></p>
<ul>
<li>
<p>ws.send() 使用连接发送数据(只能发送纯文本数据)</p>
</li>
<li>
<p>ws.close() 关闭连接</p>
</li>
</ul>
<h2 id="4demo演示">4、Demo演示</h2>
<p> 了解WebSocket的一些API之后,趁热打铁,做一个小案例跑一下。</p>
<p><strong>4.1.Node服务器端</strong></p>
<p>WebSocket协议与Node一起用非常好,原因有以下两点:</p>
<p>1.WebSocket客户端基于事件编程与Node中自定义事件差不多。</p>
<p>2.WebSocket实现客户端与服务器端长连接,Node基本事件驱动的方式十分适合高并发连接</p>
<p>创建一个webSocket.js如下:</p>
<pre><code>const WebSocketServer = require('ws').Server;
const wss = new WebSocketServer({ port: 8080 });
wss.on('connection', function (ws) {
console.log('client connected');
ws.on('message', function (message) {
ws.send('我收到了' + message);
});
});
</code></pre>
<p>打开windows命令窗口运行</p>
<p><img src="https://img2018.cnblogs.com/blog/1103967/201911/1103967-20191122022622029-1771959083.png" alt="" loading="lazy"></p>
<p><strong>4.2.HTML客户端</strong></p>
<p>新建一个index.html页面</p>
<pre><code><!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>webSocket小Demo</title>
</head>
<body>
<div class="container">
<div>
<input type="text" id="msg">
<button onclick="sendMsg()">发送报文</button>
</div>
</div>
<script>
const ws = new WebSocket('ws://localhost:8080');
ws.onmessage = (res) => {
console.log(res);
};
ws.onopen = () => {
console.log('OPEN...');
};
ws.onclose = () => {
console.log('CLOSE...');
}
function sendMsg() {
let msg = document.getElementById('msg').value;
ws.send(msg);
}
</script>
</body>
</code></pre>
<p>打开浏览器依次输入字符1,2,3,每次输入完点击发送报体,可见在ws.onmessage事件中res.data中返回来我们发的报文</p>
<p><img src="https://img2018.cnblogs.com/blog/1103967/201911/1103967-20191122022609323-1300599992.png" alt="" loading="lazy"></p>
<h2 id="5问题与总结">5、问题与总结</h2>
<p> 以上只是简单的介绍了下WebSocket的API与简单用法,在处理高并发,长连接这些需求上,例如聊天室,可能WebSocket的http请求更加合适高效。</p>
<p> 但在使用WebSocket过程中发现容易断开连接等问题,所以在每次发送报文前要判断是否断开,当多次发送报文时,由于服务器端返回数据量不同,返回客户端前后顺序也不同,所以需要在客户端收到上一个报文返回数据后再发送下一个报文,为了避免回调嵌套过多,通过Promise ,async ,await等同步方式解决。关于WebSocket就写这么多,如有不足,欢迎多多指正!</p>
<hr>
<p>参考资料:<br>
《JavaScript高级教程》<br>
《深入检出NodeJs》<br>
《菜鸟教程》</p><br><br>
来源:https://www.cnblogs.com/peerless1029/p/11909446.html
頁:
[1]