websocekt 協議
試想,如果要是想實現一個全站通知的功能,管理員在網站輸入一個通知消息,服務器接收
到這個管理員發送的消息,將這個消息來廣播出去。
傳統的實現方式是什么呢?
由于在http中 一個request 對應一個 response ,這個response是被動的。
ajax輪訓, 或者 long poll
ajax輪訓是客戶端每隔一段時間就向服務器發送一個請求,查看是否有新的通知消息。
是一種同步非阻塞的模型。
還有一種方式是long poll,如果沒有就一直等待,是一種阻塞的模型。
這兩種方式都是被動的從服務端來獲取消息。這樣會浪費很多的資源。
而websocket 能夠進行主動推送信息。經過一次http握手就可以進行。當服務端有信息
時可以直接進行推送。
HTTP/1.1 101 Switching Protocols
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Accept: HSmrc0sMlYUkAGmm5OPpG2HaGWk=
Sec-WebSocket-Protocol: chat
Upgrade: websocket
Connection: Upgrade
可以看到 這里首先來選擇協議,經過http握手后變為了websocket。
此時在瀏覽器中的抓包已經看不到websocket 的協議的內容了。
一個簡單的demo
web服務
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服務
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 的異步模式,可以進行大規模的鏈接。
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
發送消息
socket.emit