0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學(xué)習在線(xiàn)課程
  • 觀(guān)看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區
會(huì )員中心
創(chuàng )作中心

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

3天內不再提示

一文詳解WebSocket協(xié)議

jf_uPRfTJDa ? 來(lái)源: 移動(dòng)Labs ? 2024-01-07 11:26 ? 次閱讀

Labs 導讀

在WebSocket出現之前,一個(gè)Web應用(即時(shí)聊天、多人協(xié)作)的客戶(hù)端和服務(wù)端之間常見(jiàn)的雙向數據交換方式有短輪詢(xún)、長(cháng)輪詢(xún)、SSE(Server-Sent Events,服務(wù)器發(fā)送事件)。這些方式在效率和網(wǎng)絡(luò )帶寬利用率方面存在諸多問(wèn)題。WebSocket協(xié)議應運而生,對外提供了簡(jiǎn)單的雙向數據傳輸能力。

作者:朱磊

單位:中國移動(dòng)智慧家庭運營(yíng)中心成都業(yè)務(wù)支持中心

Part 01介紹

WebSocket是一種在TCP連接上進(jìn)行全雙工通訊的網(wǎng)絡(luò )通信協(xié)議。在2009年誕生,于2011年被IETF(The Internet Engineering Task Force,國際互聯(lián)網(wǎng)工程任務(wù)組)定為標準并發(fā)布RFC 6455互聯(lián)網(wǎng)標準跟蹤文檔,2016發(fā)布了RFC7936文檔進(jìn)行補充。WebSocket API同時(shí)也被W3C定為標準。

17c48bc0-ac77-11ee-8b88-92fbcf53809c.png

WebSocket協(xié)議設計之初是為了取代HTTP形式的通信,因為RFC6202中提到HTTP協(xié)議最初不是用來(lái)做雙向數據通信的。WebSocket協(xié)議并沒(méi)有完全舍棄HTTP,它基于HTTP基礎服務(wù)在現有環(huán)境中實(shí)現了雙向通信目標。正如RFC 6455中說(shuō)的那樣,WebSocke的設計哲學(xué)是最小約束的框架,唯一的約束就是協(xié)議是基于幀而不是流,并且支持Unicode文本和二進(jìn)制幀兩者。

Part 02握手

WebSocket協(xié)議分為建連握手、消息傳輸和斷連握手三個(gè)部分,整體流程如下圖所示。

17e18fe0-ac77-11ee-8b88-92fbcf53809c.png

2.1 建連握手-客戶(hù)端

為了兼容HTTP服務(wù)器側的應用程序和代理,客戶(hù)端的建連握手(包括通過(guò)代理或通過(guò)TLS加密隧道進(jìn)行的連接)是一個(gè)遵循RFC2616中定義的有效HTTP升級請求,客戶(hù)端連接握手請求header部分字段如下圖所示。此外,客戶(hù)端一旦發(fā)送了的連接握手就必須等待來(lái)自服務(wù)器的響應。

17f426f0-ac77-11ee-8b88-92fbcf53809c.png

- 請求URI

格式,ws-URI = "ws:" "http://" host [ ":" port ] path [ "?" query ]或者wss-URI = "wss:" "http://" host [ ":" port ] path [ "?" query ],任何無(wú)效值都會(huì )造成建連失敗

- 請求行

必須是GET方法,HTTP版本至少是1.1

- Upgrade

值必須是“websocket”,ASCII值,不區分大小寫(xiě)

- Connection

值必須包含“Upgrade”,ASCII值,不區分大小寫(xiě)

- Sec-WebSocket-Key

客戶(hù)端為本次建連隨機生成的16字節base64編碼的字符串

- Origin

源地址,瀏覽器客戶(hù)端必填,非瀏覽器客戶(hù)端選填

- Sec-WebSocket-Protocol

客戶(hù)端支持的一個(gè)或多個(gè)以逗號分隔的子協(xié)議,按優(yōu)先級排序

- Sec-WebSocket-Version

客戶(hù)端擬使用協(xié)議版本號,值必須為13。歷史版本9、10、11和12不再作為有效值

- Sec-WebSocket-Extensions

客戶(hù)端擬使用協(xié)議擴展。目前HyBi Working Group進(jìn)行了多路復用擴展和壓縮擴展,多路復用擴展實(shí)現共享底層TCP連接。壓縮擴展為WebSocket協(xié)議增加了壓縮功能,例如 x-webkit-deflate-frame

2.2 建連握手-服務(wù)端

當客戶(hù)端與服務(wù)端建立WebSocket連接時(shí),服務(wù)端必須回復客戶(hù)端建連握手請求,握手回復header部分字段如下圖所示。

1801bbf8-ac77-11ee-8b88-92fbcf53809c.png

- 狀態(tài)行

HTTP/1.1 101 Switching Protocols,表示接受客戶(hù)端建連。若服務(wù)器想要停止處理客戶(hù)端的握手,可返回例如401這樣的錯誤代碼的HTTP響應

- Upgrade

值必須是“websocket”

- Connection

值必須包含“Upgrade”

- Sec-WebSocket-Accept

若服務(wù)端接受客戶(hù)端連接,生成該值。先將客戶(hù)端請求頭的 Sec-WebSocket-Key值與RFC4122文檔中定義的全局唯一標識“258EAFA5-E914-47DA-95CA-C5AB0DC85B11”拼接,然后進(jìn)行SHA-1哈希再進(jìn)行base64-encoded得到該值

- Sec-WebSocket-Protocol

服務(wù)端擬使用的協(xié)議,該值從客戶(hù)端發(fā)送的Sec-WebSocket-Protocol中選擇,若服務(wù)端都不支持,值為空

- Sec-WebSocket-Extensions

服務(wù)端擬使用協(xié)議擴展

2.3 斷接握手

客戶(hù)端和服務(wù)端都可以發(fā)送包含指定控制序列的控制幀(Close控制幀)以開(kāi)始關(guān)閉握手。一方在接收到關(guān)閉控制幀時(shí),只需發(fā)送一個(gè)關(guān)閉幀作為響應,然后關(guān)閉連接。存在攔截代理等情況下,TCP關(guān)閉握手并不總是可靠的端到端握手,上述關(guān)閉握手過(guò)程旨在補充TCP關(guān)閉握手(FIN/ACK)。

Part 03數據傳輸

客戶(hù)端一旦和服務(wù)端連接握手成功,雙方就可以開(kāi)始數據傳輸了。這是一個(gè)雙向通信信道,在遵循RFC 6455規范中“消息”概念的基礎上,雙方均可以獨立地隨意發(fā)送數據。一條消息包含一個(gè)或者多個(gè)數據幀(不一定對應于網(wǎng)絡(luò )層中的消息),Websocket幀格式如下圖所示。

18130eee-ac77-11ee-8b88-92fbcf53809c.png

3.1 幀結構

- FIN

1位,表示是否是一條消息的最后一個(gè)分片。

- RSV1, RSV2, RSV3

1位,擴展功能未使用的情況下默認值為0。

- Opcode

4位,定義“Playload data”數據類(lèi)型。

0(十進(jìn)制):連續幀

1:文本幀

2:二進(jìn)制幀

3-7:預留非控制幀

8:連接關(guān)閉幀

9:心跳ping幀

10:心跳pong幀

11-15:預留控制幀

- MASK

1位,是否屏蔽“Playload data”,1是,0否。

- Payload length

7位,或者7+16位,或者7+64位,表示Payload data的長(cháng)度。具體地,Payload length小于125,數據長(cháng)度用Payload length表示;Payload length等于126,數據長(cháng)度用Payload length后面16位表示;Payload length等于127,數據長(cháng)度用Payload length后面64位表示。

- Masking-key

32位,存放客戶(hù)端發(fā)送的掩碼。為了防止代理緩存污染攻擊,RFC6455中要求掩碼必須來(lái)自強大的熵源,不可被預測。常規算法以字節為步長(cháng)遍歷載荷數據, 對于載荷數據的第i個(gè)字節, 做i對4取模得到j(luò ),掩碼覆蓋后的載荷數據的第i個(gè)字節的值為原第i個(gè)字節與Masking-Key的第j個(gè)字節做按位異或操作。

- Payload data

載荷數據分為擴展數據和應用數據兩種,擴展數據在握手階段協(xié)商是否使用,應用數據在擴展數據之后。

3.2 控制幀

控制幀由Opcode值確定,協(xié)議當前定義的控制幀的操作碼包括 0x8 (Close)、0x9(Ping)、和0xA(Pong)??刂茙仨氂幸粋€(gè)小于等于125字節的有效載荷長(cháng)度,對于Close控制幀有效載荷的前2個(gè)字節表示狀態(tài)碼,剩余字節表示關(guān)閉原因,如下圖所示。

18207d72-ac77-11ee-8b88-92fbcf53809c.png

3.3 消息分片

消息分片指將概念上的一條“消息”通過(guò)多個(gè)數據幀發(fā)送。消息分片允許發(fā)送未知大小的消息,而不必緩沖整條消息。同時(shí),消息分片結合多路復用協(xié)議的擴展,可以分割消息為更小的分段以共享輸出通道。

協(xié)議中分片消息開(kāi)始幀的FIN位為0,opcode位為非0表示該幀為某消息分片,中間幀FIN位為0,opcode位為0,最后通過(guò)FIN位為1,opcode為0標識分片結束。協(xié)議要求分片數據幀按順序發(fā)送到另一端。

Part 04總結

WebSocke是設計在TCP層之上,不需要考慮數據長(cháng)度,數據粘包拆包。也能通過(guò)擴展功能與HTTP/2多路復用結合,充分利用帶寬。開(kāi)發(fā)者只需在服務(wù)端和客戶(hù)端代碼中按序處理消息分片邏輯。

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權轉載。文章觀(guān)點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習之用,如有內容侵權或者其他違規問(wèn)題,請聯(lián)系本站處理。 舉報投訴
  • 服務(wù)器
    +關(guān)注

    關(guān)注

    12

    文章

    8302

    瀏覽量

    83229
  • 網(wǎng)絡(luò )通信

    關(guān)注

    4

    文章

    737

    瀏覽量

    29597
  • 網(wǎng)絡(luò )帶寬

    關(guān)注

    0

    文章

    33

    瀏覽量

    8120
  • WebSocket
    +關(guān)注

    關(guān)注

    0

    文章

    24

    瀏覽量

    3677

原文標題:技術(shù) | 淺談WebSocket協(xié)議-RFC 6455

文章出處:【微信號:5G通信,微信公眾號:5G通信】歡迎添加關(guān)注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    Django3如何使用WebSocket實(shí)現WebShell

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

    NetRouter開(kāi)發(fā)板之Websocket

    何為websocket協(xié)議,可以理解介乎http和tcp/ip之間為啥要用他為了實(shí)現服務(wù)器向客戶(hù)端推送信息。對于嵌入式來(lái)說(shuō)。就是用戶(hù)頁(yè)面,要實(shí)時(shí)展示前端獲取信息。比如,用瀏覽器實(shí)時(shí)顯示室內溫度
    發(fā)表于 06-20 20:45

    網(wǎng)頁(yè)控制設備 Websocket API

    綁定列表如果該用戶(hù)要獲取綁定設備列表,先要確認該用戶(hù)(就是該openid)是否有對要控制設備的已進(jìn)行綁定了。 1.3.創(chuàng )建Websocket連接選擇個(gè)要進(jìn)行控制的設備did,創(chuàng )建websocket
    發(fā)表于 01-23 15:58

    讀懂什么是NEC協(xié)議

    讀懂什么是NEC協(xié)議?
    發(fā)表于 10-15 09:22

    基于TCP的種新的網(wǎng)絡(luò )協(xié)議WebSocket

    開(kāi)啟 WebSocket 服務(wù)WebSocket 服務(wù)是網(wǎng)頁(yè)程序、安卓 App、微信小程序等獲得數據和服務(wù)的接口,是基于TCP 的種新的網(wǎng)絡(luò )協(xié)議,它實(shí)現了瀏覽器與服務(wù)器全雙工通信。通
    發(fā)表于 12-16 07:38

    Modbus_通訊協(xié)議詳解

    Modbus_通訊協(xié)議詳解,Modbus_通訊協(xié)議詳解
    發(fā)表于 12-08 14:13 ?0次下載

    什么是WebSocket?進(jìn)行通信解析 WebSocket 報文及實(shí)現

    一般情況下全為 0。當客戶(hù)端、服務(wù)端協(xié)商采用 WebSocket 擴展時(shí),這三個(gè)標志位可以非0,且值的含義由擴展進(jìn)行定義。如果出現非零的值,且并沒(méi)有采用 WebSocket 擴展,連接出錯。
    的頭像 發(fā)表于 05-15 16:59 ?9328次閱讀
    什么是<b class='flag-5'>WebSocket</b>?進(jìn)行通信解析 <b class='flag-5'>WebSocket</b> 報文及實(shí)現

    根據WebSocket協(xié)議完全使用C++實(shí)現函數

    由于需要在項目中增加Websocket協(xié)議,與客戶(hù)端進(jìn)行通信,不想使用開(kāi)源的庫,比如WebSocketPP,就自己根據WebSocket協(xié)議實(shí)現一套函數,完全使用C++實(shí)現。
    的頭像 發(fā)表于 11-28 14:29 ?4303次閱讀

    Python如何爬取實(shí)時(shí)變化的WebSocket數據

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

    WebSocket有什么優(yōu)點(diǎn)

    WebSocket是一種在單個(gè)TCP連接上進(jìn)行全雙工通信的協(xié)議。WebSocket通信協(xié)議于2011年被IETF定為標準RFC 6455,并由RFC7936補充規范。
    的頭像 發(fā)表于 02-15 15:53 ?8031次閱讀
    <b class='flag-5'>WebSocket</b>有什么優(yōu)點(diǎn)

    WebSocket工作原理及使用方法

    它有很多名字; WebSocket,WebSocket協(xié)議WebSocket API。從首選的消息傳遞應用程序到流行的在線(xiàn)多人游戲,WebSock
    的頭像 發(fā)表于 05-05 22:12 ?7619次閱讀
    <b class='flag-5'>WebSocket</b>工作原理及使用方法

    FreeSwitch的sip協(xié)議協(xié)議詳解.pdf

    FreeSwitch的sip協(xié)議協(xié)議詳解.pdf
    發(fā)表于 12-30 11:28 ?3次下載

    ESP32 單片機學(xué)習筆記 - 08 - WebSocket客戶(hù)端

    前言,終于要到網(wǎng)絡(luò )模型的最后一層,第四層,應用層,http、websocket的實(shí)踐了。文章目錄ESP32 單片機學(xué)習筆記 - 08 - WebSocket客戶(hù)端一、應用層協(xié)議 科普概念二、編程指南
    發(fā)表于 12-29 18:56 ?9次下載
    ESP32 單片機學(xué)習筆記 - 08 - <b class='flag-5'>WebSocket</b>客戶(hù)端

    鴻蒙上WebSocket的使用方法

    WebSocket 是一種網(wǎng)絡(luò )通訊協(xié)議,很多網(wǎng)絡(luò )開(kāi)發(fā)工作者都需要它。本文介紹在 OpenHarmony 上 WebSocket 協(xié)議的使用方法。
    的頭像 發(fā)表于 03-08 14:17 ?1152次閱讀

    websocket協(xié)議的原理

    WebSocket協(xié)議是基于TCP的一種新的網(wǎng)絡(luò )協(xié)議。它實(shí)現了瀏覽器與服務(wù)器全雙工(full-duplex)通信——允許服務(wù)器主動(dòng)發(fā)送信息給客戶(hù)端。 WebSocket通信
    的頭像 發(fā)表于 11-09 15:13 ?520次閱讀
    <b class='flag-5'>websocket</b><b class='flag-5'>協(xié)議</b>的原理
    亚洲欧美日韩精品久久_久久精品AⅤ无码中文_日本中文字幕有码在线播放_亚洲视频高清不卡在线观看