python 用struct模塊解決黏包問(wèn)題
為什么會(huì)出現(xiàn)黏包現(xiàn)象:
首先只有在TCP協(xié)議中才會(huì)出現(xiàn)黏包現(xiàn)象,是因?yàn)門CP協(xié)議是面向流的協(xié)議,在發(fā)送的數(shù)據(jù)傳輸?shù)倪^(guò)程中還有緩存機(jī)制來(lái)避免數(shù)據(jù)丟失,因此,在連續(xù)發(fā)送小數(shù)據(jù)的時(shí)候,以及接收大小不符的時(shí)候容易出現(xiàn)黏包現(xiàn)象。本質(zhì)還是因?yàn)槲覀冊(cè)诮邮諗?shù)據(jù)的時(shí)候不知道發(fā)送的數(shù)據(jù)的長(zhǎng)短。
解決黏包問(wèn)題
在傳輸大量數(shù)據(jù)之前首先告訴接收端要發(fā)送的數(shù)據(jù)大小,如果想更漂亮的解決問(wèn)題,可以通過(guò)struct模塊來(lái)定制協(xié)議。
struct模塊:
功能:可以把一個(gè)類型,如數(shù)字,轉(zhuǎn)成固定長(zhǎng)度的bytes。
import struct
ret = struct.pack('i',456872783) #'i'代表int,就是即將要把一共數(shù)字轉(zhuǎn)換成固定長(zhǎng)度(4個(gè)字節(jié))的bystes類型
print(ret)
num = struct.unpack('i',ret) #轉(zhuǎn)換回來(lái),返回一個(gè)元組
print(num[0]) #提前元組中的值得到4096
解決黏包問(wèn)題:
服務(wù)端:
import struct
import socket
sk = socket.socket()
sk.bind(('127.0.0.1',8080))
sk.listen()
conn,addr = sk.accept()
while True:
cmd = input('>>>')
if cmd == 'q': #當(dāng)輸入‘q'時(shí),結(jié)束,并向客戶端發(fā)送一個(gè)'q'。
conn.send(b'q')
break
conn.send(cmd.encode('gbk')) #將輸入的cmd命令發(fā)送給客戶端
num = conn.recv(4) #接收字節(jié)信息(返回的消息長(zhǎng)度信息)。
num = struct.unpack('i',num)[0] #將接收的字節(jié)碼轉(zhuǎn)化為原來(lái)的類型并放在一個(gè)元組里面,后面加[0]是提前出元組中的值。
res = conn.recv(int(num)).decode('gbk') #接收長(zhǎng)度為num 的消息。
print(res) #打印
conn.close()
sk.close()
客戶端:
import struct
import socket
import subprocess
sk = socket.socket()
sk.connect(('127.0.0.1',8080))
while True:
cmd = sk.recv(1024).decode('gbk') #接收服務(wù)端發(fā)送來(lái)的cmd命令
if cmd == 'q': #當(dāng)接收到‘q'時(shí),結(jié)束。
break
# 在客戶端執(zhí)行接收到的cmd命令。并將正確的消息和錯(cuò)誤的消息分別放入stdout和stderr管道。
res = subprocess.Popen(cmd,shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
std_out = res.stdout.read() #讀取管道內(nèi)正確的消息
std_err = res.stderr.read() #讀取管道內(nèi)錯(cuò)誤的消息
len_num = len(std_out)+len(std_err) #計(jì)算正確和錯(cuò)誤消息的總長(zhǎng)度
num_by = struct.pack('i',len_num) #將消息總長(zhǎng)度轉(zhuǎn)換成長(zhǎng)度為4的字節(jié)碼
sk.send(num_by) #發(fā)送消息長(zhǎng)度信息
sk.send(std_out) #發(fā)送正確消息
sk.send(std_err) #發(fā)送錯(cuò)誤消息
sk.close()
以上就是python 用struct模塊解決黏包問(wèn)題的詳細(xì)內(nèi)容,更多關(guān)于python struct模塊的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
python3.6+selenium實(shí)現(xiàn)操作Frame中的頁(yè)面元素
這篇文章主要為大家詳細(xì)介紹了python3.6+selenium實(shí)現(xiàn)操作Frame中的頁(yè)面元素,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-07-07
Python decorator攔截器代碼實(shí)例解析
這篇文章主要介紹了Python decorator攔截器代碼實(shí)例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04
python實(shí)現(xiàn)百萬(wàn)答題自動(dòng)百度搜索答案
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)百萬(wàn)答題自動(dòng)百度搜索答案,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01
實(shí)例講解Python中g(shù)lobal語(yǔ)句下全局變量的值的修改
global是Python中的一個(gè)關(guān)鍵字用來(lái),聲明一個(gè)局部變量為全局變量,這里我們來(lái)以實(shí)例講解Python中g(shù)lobal語(yǔ)句下全局變量的值的修改,需要的朋友可以參考下.2016-06-06
Python手繪可視化工具cutecharts使用實(shí)例
這篇文章主要介紹了Python手繪可視化工具cutecharts使用實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
Python之try無(wú)法使用全局變量的問(wèn)題解決
當(dāng)我們使用try語(yǔ)句時(shí),如果在try中使用了全局變量,但又在except或finally中修改了這個(gè)全局變量,就會(huì)出現(xiàn)這種無(wú)法修改全局變量的情況,下面就來(lái)解決一下這個(gè)問(wèn)題,感興趣的可以了解一下2024-08-08

