python設計tcp數(shù)據(jù)包協(xié)議類的例子
一. 問題描述
在tcp編程中,最需要解決的就是粘包分包問題。所以,我們需要在每個數(shù)據(jù)包前面加上數(shù)據(jù)包的長度用以分割粘連的包。
二. 包結構的設計
包的組成:包長度+數(shù)據(jù)域
包長度:用4個字節(jié)存儲數(shù)據(jù)域長度,數(shù)據(jù)域長度即為其所占字節(jié)數(shù)
數(shù)據(jù)域:由若干個變量組成,如果是定長變量則不用加變量長度
定長變量:我們人為規(guī)定,傳輸中的int為4字節(jié)定長變量
變長變量:那就是字符串啦
文字難理解,那我就畫個圖吧:
上圖的第一行是數(shù)據(jù)包的一個總體結構
第二行是數(shù)據(jù)域內部的一個結構(數(shù)據(jù)域的變量數(shù)量和位置都是我們自己定的,上圖只是舉一個例子而已)
第三行是具體變量的結構
如果不太清楚這個結構,不要緊,我們來舉一個具體的例子
比如我們現(xiàn)在創(chuàng)建一個數(shù)據(jù)域是這樣的數(shù)據(jù)包:
數(shù)據(jù)域:666,"你好啊","hello",888
這個數(shù)據(jù)域一共存儲了四個變量,開頭和結尾是兩個整型變量,中間是兩個字符串變量。然后我們對這個數(shù)據(jù)域構建出來的數(shù)據(jù)包是這個樣子的:
這下搞明白了吧,那下面就看看怎么用python封裝一個類實現(xiàn)上述結構的數(shù)據(jù)包的組裝。
三. 代碼實現(xiàn)
class Protocol: """ 規(guī)定: 數(shù)據(jù)包頭部占4字節(jié) 整型占4字節(jié) 字符串長度位占2字節(jié) 字符串不定長 """ def __init__(self, bs=None): """ 如果bs為None則代表需要創(chuàng)建一個數(shù)據(jù)包 否則代表需要解析一個數(shù)據(jù)包 """ if bs: self.bs = bytearray(bs) else: self.bs = bytearray(0) def get_int32(self): try: ret = self.bs[:4] self.bs = self.bs[4:] return int.from_bytes(ret, byteorder='little') except: raise Exception("數(shù)據(jù)異常!") def get_str(self): try: # 拿到字符串字節(jié)長度(字符串長度位2字節(jié)) length = int.from_bytes(self.bs[:2], byteorder='little') # 再拿字符串 ret = self.bs[2:length + 2] # 刪掉取出來的部分 self.bs = self.bs[2 + length:] return ret.decode(encoding='utf8') except: raise Exception("數(shù)據(jù)異常!") def add_int32(self, val): bytes_val = bytearray(val.to_bytes(4, byteorder='little')) self.bs += bytes_val def add_str(self, val): bytes_val = bytearray(val.encode(encoding='utf8')) bytes_length = bytearray(len(bytes_val).to_bytes(2, byteorder='little')) self.bs += (bytes_length + bytes_val) def get_pck_not_head(self): return self.bs def get_pck_has_head(self): bytes_pck_length = bytearray(len(self.bs).to_bytes(4, byteorder='little')) return bytes_pck_length + self.bs if __name__ == '__main__': p = Protocol() p.add_int32(666) p.add_str("你好啊") p.add_str("hello") p.add_int32(888) r = Protocol(p.get_pck_not_head()) print(r.get_int32()) print(r.get_str()) print(r.get_str()) print(r.get_int32())
代碼比較簡單,也不夠嚴謹。大家可以按照自己的需求加以修改。
以上這篇python設計tcp數(shù)據(jù)包協(xié)議類的例子就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
python數(shù)據(jù)分析之如何刪除value=0的行
這篇文章主要給大家介紹了關于python數(shù)據(jù)分析之如何刪除value=0的行的相關資料,文中通過實例代碼以及圖文介紹的非常詳細,對大家學習或者使用Python具有一定的參考學習價值,需要的朋友可以參考下2022-12-12Python基于Socket實現(xiàn)簡易多人聊天室的示例代碼
這篇文章主要介紹了Python基于Socket實現(xiàn)簡易多人聊天室的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-11-11在vscode中啟動conda虛擬環(huán)境的思路詳解
這篇文章主要介紹了在vscode中啟動conda虛擬環(huán)境的思路詳解,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-12-12pycharm創(chuàng)建并使用虛擬環(huán)境的詳細圖文教程
Python的虛擬環(huán)境是正常的現(xiàn)實環(huán)境相對應的,在虛擬環(huán)境中安裝的包是與現(xiàn)實環(huán)境隔離的,下面這篇文章主要給大家介紹了關于pycharm創(chuàng)建并使用虛擬環(huán)境的詳細圖文教程,需要的朋友可以參考下2022-08-08基于python的selenium兩種文件上傳操作實現(xiàn)詳解
這篇文章主要介紹了基于python的selenium兩種文件上傳操作實現(xiàn)詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2019-09-09