websocekt 協(xié)議
試想,如果要是想實現(xiàn)一個全站通知的功能,管理員在網(wǎng)站輸入一個通知消息,服務(wù)器接收
到這個管理員發(fā)送的消息,將這個消息來廣播出去。
傳統(tǒng)的實現(xiàn)方式是什么呢?
由于在http中 一個request 對應(yīng)一個 response ,這個response是被動的。
ajax輪訓(xùn), 或者 long poll
ajax輪訓(xùn)是客戶端每隔一段時間就向服務(wù)器發(fā)送一個請求,查看是否有新的通知消息。
是一種同步非阻塞的模型。
還有一種方式是long poll,如果沒有就一直等待,是一種阻塞的模型。
這兩種方式都是被動的從服務(wù)端來獲取消息。這樣會浪費很多的資源。
而websocket 能夠進行主動推送信息。經(jīng)過一次http握手就可以進行。當(dāng)服務(wù)端有信息
時可以直接進行推送。
HTTP/1.1 101 Switching Protocols
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Accept: HSmrc0sMlYUkAGmm5OPpG2HaGWk=
Sec-WebSocket-Protocol: chat
Upgrade: websocket
Connection: Upgrade
可以看到 這里首先來選擇協(xié)議,經(jīng)過http握手后變?yōu)榱藈ebsocket。
此時在瀏覽器中的抓包已經(jīng)看不到websocket 的協(xié)議的內(nèi)容了。
一個簡單的demo
web服務(wù)
from flask import Flask, render_template
app = Flask(__name__)
@app.route('/')
def index():
return render_template('socket.html')
if __name__ == '__main__':
app.run(debug=True)
socketio服務(wù)
from aiohttp import web
import socketio
sio = socketio.AsyncServer(async_mode='aiohttp')
app = web.Application()
sio.attach(app)
@sio.on('my broadcast event', namespace='/test')
async def test_broadcast_message(sid, message):
await sio.emit('my response', {'data': message['data']}, namespace='/test')
if __name__ == '__main__':
web.run_app(app, host='localhost', port=8000)
上面使用了asyncio 的異步模式,可以進行大規(guī)模的鏈接。
socket.html
主要如下:
引入socketio 包,這個包可以在不支持websocket的瀏覽器上切換成其他方式
<script type="text/javascript" src="http://cdnjs.cloudflare.com/ajax/libs/socket.io/1.3.5/socket.io.min.js"></script>
實例化
var socket = io.connect('http://' + document.domain + ':' + port+ namespace);
接收消息
socket.on
發(fā)送消息
socket.emit