您好,登錄后才能下訂單哦!
小編給大家分享一下如何解決Python socket連接中的粘包、精確傳輸問題,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!
當(dāng)調(diào)用send的時候,數(shù)據(jù)并不是即時發(fā)給客戶端的。而是放到了系統(tǒng)的socket發(fā)送緩沖區(qū)里,等緩沖區(qū)滿了、或者數(shù)據(jù)等待超時了,數(shù)據(jù)才會發(fā)送,所以有時候發(fā)送太快的話,前一份數(shù)據(jù)還沒有傳給客戶端,那么這份數(shù)據(jù)和上一份數(shù)據(jù)一起發(fā)給客戶端的時候就會造成“粘包” 。
解決根源的思想是避免不同段的數(shù)據(jù)一起發(fā)送。
方案1:前一段數(shù)據(jù)send完后,等待一段時間再send第二段數(shù)據(jù)。缺點:時間效率低,而且也無法完全避免問題【因為不清楚該設(shè)置多少時間才能保證前一份數(shù)據(jù)已經(jīng)發(fā)送】
方案2:握手機制:前一段數(shù)據(jù)send完后,嘗試recv,等待客戶端回應(yīng),確認(rèn)第一段數(shù)據(jù)發(fā)送完后,再send第二段數(shù)據(jù)。完美方案?
服務(wù)端【發(fā)送方】代碼:
import socket server=socket.socket() server.bind(("localhost",1234)) server.listen() while True: print("正在等待。。。") conn,addr=server.accept() while True: try: conn.send(b"first info") ack=conn.recv(1024) #接收客戶端確認(rèn) print(ack) conn.send(b"second info") except ConnectionResetError as e: print(e) break server.close()
客戶端【接收方】代碼:
import socket client=socket.socket() client.connect(("localhost",1234)) data=client.recv(1024) print(data.decode()) client.send(b"ack")#發(fā)送確認(rèn) data=client.recv(1024) print(data.decode()) client.close()
由于數(shù)據(jù)太大,發(fā)送方一次send不完,而接收方只recv一次,使得影響了后面數(shù)據(jù)的傳輸
解決根源的思想是改變recv的次數(shù)。
方案:將數(shù)據(jù)的大小發(fā)給接收方,讓接收方來決定recv的次數(shù)
服務(wù)端【發(fā)送方】:
import socket,os server=socket.socket() server.bind(("localhost",1234)) server.listen() while True: print("正在等待...") conn,addr=server.accept() print("連接成功!") while True: try: cmd=conn.recv(1024) data=os.popen(cmd.decode()).read() # print(data) cmd_len=len(data.encode()) print(cmd_len) #發(fā)現(xiàn)這里如果cmd_len為0會導(dǎo)致異常,有些是沒有返回值的command if cmd_len==0: data="command has nothing return" cmd_len=len(data.encode()) ##因為這里前面沒有發(fā)送操作,所以不用擔(dān)心粘包,如果有則要考慮處理 conn.send(str(cmd_len).encode())#因為len結(jié)果是int,所以還要轉(zhuǎn)換 #這里要處理粘包 ack=conn.recv(1024) conn.send(data.encode()) except ConnectionResetError as e: print(e) break server.close()
客戶端【接收方】:
import socket client=socket.socket() client.connect(("localhost",1234)) while True: cmd = input(">>:") client.send(cmd.encode()) data_len=client.recv(1024) data_len=int(data_len.decode()) print(data_len) recv_len=0 client.send(b'ack') total_data=b'' while recv_len<data_len: data=client.recv(1024) recv_len+=len(data) total_data+=data print(total_data.decode()) client.close()
利用這個原理可以實現(xiàn)文件傳輸,只要能確定接受次數(shù),就能保證文件傳輸?shù)拇笮≌_。
看完了這篇文章,相信你對“如何解決Python socket連接中的粘包、精確傳輸問題”有了一定的了解,如果想了解更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。