精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

什么是WebSocket?進行通信解析 WebSocket 報文及實現

馬哥Linux運維 ? 來源:未知 ? 作者:李倩 ? 2018-05-15 16:59 ? 次閱讀

什么是 WebSocket ?

WebSocket 是一種標準協議,用于在客戶端和服務端之間進行雙向數據傳輸。但它跟 HTTP 沒什么關系,它是一種基于 TCP 的一種獨立實現。

以前客戶端想知道服務端的處理進度,要不停地使用 Ajax 進行輪詢,讓瀏覽器隔個幾秒就向服務器發一次請求,這對服務器壓力較高。另外一種輪詢就是采用 long poll 的方式,這就跟打電話差不多,沒收到消息就一直不掛電話,也就是說,客戶端發起連接后,如果沒消息,就一直不返回 Response 給客戶端,連接階段一直是阻塞的。

而 WebSocket 解決了 HTTP 的這幾個難題。首先,當服務器完成協議升級后( HTTP -> WebSocket ),服務端可以主動推送信息給客戶端,解決了輪詢造成的同步延遲問題。由于 WebSocket 只需要一次 HTTP 握手,服務端就能一直與客戶端保持通訊,直到關閉連接,這樣就解決了服務器需要反復解析 HTTP 協議,減少了資源的開銷。

隨著新標準的推進,WebSocket 已經比較成熟了,并且各個瀏覽器對 WebSocket 的支持情況比較好,有空可以看看。

使用 WebSocket 的時候,前端使用是比較規范的,js 支持 ws 協議,感覺類似于一個輕度封裝的 Socket 協議,只是以前需要自己維護 Socket 的連接,現在能夠以比較標準的方法來進行。

客戶端請求報文及實現

客戶端請求報文:

GET / HTTP/1.1Upgrade: websocketConnection: UpgradeHost: example.comOrigin: http://example.comSec-WebSocket-Key: sN9cRrP/n9NdMgdcy2VJFQ==Sec-WebSocket-Version: 13

與傳統 HTTP 報文不同的地方:

Upgrade: websocket Connection: Upgrade

這兩行表示發起的是 WebSocket 協議。

Sec-WebSocket-Key: sN9cRrP/n9NdMgdcy2VJFQ==Sec-WebSocket-Version: 13

Sec-WebSocket-Key是由瀏覽器隨機生成的,提供基本的防護,防止惡意或者無意的連接。

Sec-WebSocket-Version表示 WebSocket 的版本,最初 WebSocket 協議太多,不同廠商都有自己的協議版本,不過現在已經定下來了。如果服務端不支持該版本,需要返回一個Sec-WebSocket-Versionheader,里面包含服務端支持的版本號。

創建 WebSocket 對象:

var ws = new websocket("ws://127.0.0.1:8001");

ws 表示使用 WebSocket 協議,后面接地址及端口

完整的客戶端代碼:

服務端響應報文及實現

首先我們來看看服務端的響應報文

HTTP/1.1 101 Switching ProtocolsUpgrade: websocketConnection: UpgradeSec-WebSocket-Accept: HSmrc0sMlYUkAGmm5OPpG2HaGWk=Sec-WebSocket-Protocol: chat

我們一行行來解釋

1、首先,101 狀態碼表示服務器已經理解了客戶端的請求,并將通過Upgrade消息頭通知客戶端采用不同的協議來完成這個請求;

2、然后,Sec-WebSocket-Accept這個則是經過服務器確認,并且加密過后的Sec-WebSocket-Key;

3、最后,Sec-WebSocket-Protocol則是表示最終使用的協議。

Sec-WebSocket-Accept的計算方法:

1、將Sec-WebSocket-Key跟 258EAFA5-E914-47DA-95CA-C5AB0DC85B11 拼接;

2、通過 SHA1 計算出摘要,并轉成 base64 字符串。

PS:Sec-WebSocket-Key/Sec-WebSocket-Accept的換算,只能帶來基本的保障,但連接是否安全、數據是否安全、客戶端 / 服務端是否合法的 ws 客戶端、ws 服務端,其實并沒有實際性的保證。

創建主線程,用于實現接受 WebSocket 建立請求:

def create_socket():
# 啟動 Socket 并監聽連接 sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) try:
sock.bind(('127.0.0.1', 8001))
# 操作系統會在服務器 Socket 被關閉或服務器進程終止后馬上釋放該服務器的端口,否則操作系統會保留幾分鐘該端口。
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.listen(5)
except Exception as e:
logging.error(e)
return
else:
logging.info('Server running...') # 等待訪問
while True:
conn, addr = sock.accept() # 此時會進入 waiting 狀態
data = str(conn.recv(1024))
logging.debug(data)
header_dict = {}
header, _ = data.split(r'\r\n\r\n', 1)
for line in header.split(r'\r\n')[1:]:
key, val = line.split(': ', 1)
header_dict[key] = val
if 'Sec-WebSocket-Key' not in header_dict:
logging.error('This socket is not websocket, client close.')
conn.close()
return
magic_key = '258EAFA5-E914-47DA-95CA-C5AB0DC85B11'
sec_key = header_dict['Sec-WebSocket-Key'] + magic_key
key = base64.b64encode(hashlib.sha1(bytes(sec_key, encoding='utf-8')).digest())
key_str = str(key)[2:30]
logging.debug(key_str)
response = 'HTTP/1.1 101 Switching Protocols\r\n' \
'Connection: Upgrade\r\n' \
'Upgrade: websocket\r\n' \
'Sec-WebSocket-Accept: {0}\r\n' \
'WebSocket-Protocol: chat\r\n\r\n'.format(key_str)
conn.send(bytes(response, encoding='utf-8'))
logging.debug('Send the handshake data')
WebSocketThread(conn).start()

進行通信解析 WebSocket 報文及實現

Server 端接收到瀏覽器發來的報文需要進行解析

瀏覽器包格式

1、FIN: 占 1 個 bit

0:不是消息的最后一個分片1:是消息的最后一個分片

2、RSV1, RSV2, RSV3:各占 1 個 bit

一般情況下全為 0。當客戶端、服務端協商采用 WebSocket 擴展時,這三個標志位可以非0,且值的含義由擴展進行定義。如果出現非零的值,且并沒有采用 WebSocket 擴展,連接出錯。

3、Opcode: 4 個 bit

%x0:表示一個延續幀。當 Opcode 為 0 時,表示本次數據傳輸采用了數據分片,當前收到的數據幀為其中一個數據分片;%x1:表示這是一個文本幀(frame);%x2:表示這是一個二進制幀(frame);%x3-7:保留的操作代碼,用于后續定義的非控制幀;%x8:表示連接斷開;%x9:表示這是一個 ping 操作;%xA:表示這是一個 pong 操作;%xB-F:保留的操作代碼,用于后續定義的控制幀。

4、Mask: 1 個 bit

表示是否要對數據載荷進行掩碼異或操作。0:否1:是

5、Payload length: 7bit or 7 + 16bit or 7 + 64bit

表示數據載荷的長度x為 0~126:數據的長度為x字節;x為 126:后續 2 個字節代表一個 16 位的無符號整數,該無符號整數的值為數據的長度;x為 127:后續 8 個字節代表一個 64 位的無符號整數(最高位為 0),該無符號整數的值為數據的長度。

6、Masking-key: 0 or 4bytes

當 Mask 為 1,則攜帶了 4 字節的 Masking-key;當 Mask 為 0,則沒有 Masking-key。PS:掩碼的作用并不是為了防止數據泄密,而是為了防止早期版本的協議中存在的代理緩存污染攻擊(proxy cache poisoning attacks)等問題。

7、Payload Data: 載荷數據

解析 WebSocket 報文代碼如下:

def read_msg(data):
logging.debug(data)
msg_len = data[1] & 127 # 數據載荷的長度
if msg_len == 126:
mask = data[4:8] # Mask 掩碼
content = data[8:] # 消息內容
elif msg_len == 127:
mask = data[10:14]
content = data[14:]
else:
mask = data[2:6]
content = data[6:]
raw_str = '' # 解碼后的內容 for i, d in enumerate(content):
raw_str += chr(d ^ mask[i % 4])
return raw_str

服務端發送 WebSocket 報文

返回時不攜帶掩碼,所以 Mask 位為 0,再按載荷數據的大小寫入長度,最后寫入載荷數據。

struct 模塊解析

struct.pack(fmt, v1, v2, ...)

按照給定的格式 (fmt),把數據封裝成字符串 ( 實際上是類似于 c 結構體的字節流 )

struct 中支持的格式如下表:

為了同 C 語言中的結構體交換數據,還要考慮有的 C 或 C++ 編譯器使用了字節對齊,通常是以 4 個字節為單位的 32 位系統,故而 struct 根據本地機器字節順序轉換。可以用格式中的第一個字符來改變對齊方式,定義如下:

發送 WebSocket 報文代碼如下:

def write_msg(message):
data = struct.pack('B', 129) # 寫入第一個字節,10000001
# 寫入包長度 msg_len = len(message) if msg_len <= 125: ? ? ?
data += struct.pack('B', msg_len)
elif msg_len <= (2 ** 16 - 1): ? ? ?
data += struct.pack('!BH', 126, msg_len) elif msg_len <= (2 ** 64 - 1): ?
data += struct.pack('!BQ', 127, msg_len)
else:
logging.error('Message is too long!')
return
data += bytes(message, encoding='utf-8')
# 寫入消息內容
logging.debug(data) return data

總結

沒有其他能像 WebSocket 一樣實現雙向通信的技術了,迄今為止,大部分開發者還是使用 Ajax 輪詢來實現,但這是個不太優雅的解決辦法,WebSocket 雖然用的人不多,可能是因為協議剛出來的時候有安全性的問題以及兼容的瀏覽器比較少,但現在都有解決。如果你有這些需求可以考慮使用 WebSocket:

1 、多個用戶之間進行交互;

2、需要頻繁地向服務端請求更新數據。

比如彈幕、消息訂閱、多玩家游戲、協同編輯、股票基金實時報價、視頻會議、在線教育等需要高實時的場景。

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 通信
    +關注

    關注

    18

    文章

    5969

    瀏覽量

    135850
  • TCP
    TCP
    +關注

    關注

    8

    文章

    1349

    瀏覽量

    78985
  • WebSocket
    +關注

    關注

    0

    文章

    29

    瀏覽量

    3737

原文標題:一文讀懂 WebSocket 通信過程與實現

文章出處:【微信號:magedu-Linux,微信公眾號:馬哥Linux運維】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    Django3如何使用WebSocket實現WebShell

    websocket 服務。 大致看了下覺得這不夠有趣,翻了翻 django 的官方文檔發現 django 原生是不支持 websocket 的,但 django3 之后支持了 asgi 協議可以自己實現
    的頭像 發表于 11-17 09:58 ?4334次閱讀

    鴻蒙原生應用開發-網絡管理WebSocket連接

    ,客戶端會收到open事件的回調,之后客戶端就可以通過send()方法與服務器進行通信。當服務器發信息給客戶端時,客戶端會收到message事件的回調。當客戶端不要此連接時,可以通過調用close
    發表于 04-07 09:46

    ESP32進行websocket通信接收數據出錯的原因?

    I (104793) WEBSOCKET: WEBSOCKET_EVENT_CONNECTED I (104953) WEBSOCKET: WEBSOCKET_EVENT_DATA I
    發表于 06-14 07:42

    如何通過Java來使用WebSocket

    很酷的新技術,可以實現瀏覽器與服務器之間實時、雙向的通信,幾乎沒有任何額外的代價。我這里要做的事情就是提供一個非常簡潔,但卻內容豐富的概覽,介紹如何開始使用這門技術。首先讀者需要了解如下一些事情:
    發表于 07-11 07:19

    初探實現websocket的心跳重連

    初探和實現websocket心跳重連
    發表于 05-25 15:07

    基于TCP的一種新的網絡協議WebSocket

    開啟 WebSocket 服務WebSocket 服務是網頁程序、安卓 App、微信小程序等獲得數據和服務的接口,是基于TCP 的一種新的網絡協議,它實現了瀏覽器與服務器全雙工通信。通
    發表于 12-16 07:38

    ESP32進行websocket通信接收數據出錯的原因?

    I (104793) WEBSOCKET: WEBSOCKET_EVENT_CONNECTEDI (104953) WEBSOCKET: WEBSOCKET_EVENT_DATAI (
    發表于 02-13 06:46

    ESP32進行websocket通信接收數據出錯怎么解決?

    I (104793) WEBSOCKET: WEBSOCKET_EVENT_CONNECTEDI (104953) WEBSOCKET: WEBSOCKET_EVENT_DATAI (
    發表于 03-07 06:59

    根據WebSocket協議完全使用C++實現函數

    由于需要在項目中增加Websocket協議,與客戶端進行通信,不想使用開源的庫,比如WebSocketPP,就自己根據WebSocket協議實現
    的頭像 發表于 11-28 14:29 ?4842次閱讀

    Python如何爬取實時變化的WebSocket數據

    Python 中的網絡請求庫非常多,Requests 是最常用的請求庫之一,它可以模擬發送網絡請求。但是這些請求都是基于 HTTP 協議的。在面對 WebSocket 的時候 Requests 就發揮不料作用了,必須使用能夠連接 WebSocket 的庫。
    的頭像 發表于 03-11 09:31 ?3538次閱讀
    Python如何爬取實時變化的<b class='flag-5'>WebSocket</b>數據

    WebSocket有什么優點

    WebSocket是一種在單個TCP連接上進行全雙工通信的協議。WebSocket通信協議于2011年被IETF定為標準RFC 6455,并
    的頭像 發表于 02-15 15:53 ?8252次閱讀
    <b class='flag-5'>WebSocket</b>有什么優點

    WebSocket工作原理及使用方法

    它有很多名字; WebSocketWebSocket協議和WebSocket API。從首選的消息傳遞應用程序到流行的在線多人游戲,WebSocket在當今最常用的Web應用程序中是
    的頭像 發表于 05-05 22:12 ?7841次閱讀
    <b class='flag-5'>WebSocket</b>工作原理及使用方法

    鴻蒙上WebSocket的使用方法

    WebSocket 是一種網絡通訊協議,很多網絡開發工作者都需要它。本文介紹在 OpenHarmony 上 WebSocket 協議的使用方法。
    的頭像 發表于 03-08 14:17 ?1796次閱讀

    WebSocket的6種集成方式介紹

    由于前段時間我實現了一個庫【Spring Cloud】一個配置注解實現 WebSocket 集群方案
    的頭像 發表于 09-02 16:52 ?1439次閱讀

    websocket協議的原理

    WebSocket協議是基于TCP的一種新的網絡協議。它實現了瀏覽器與服務器全雙工(full-duplex)通信——允許服務器主動發送信息給客戶端。 WebSocket
    的頭像 發表于 11-09 15:13 ?1111次閱讀
    <b class='flag-5'>websocket</b>協議的原理