Python socket連接中的粘包、精確傳輸問(wèn)題實(shí)例分析
本文實(shí)例講述了Python socket連接中的粘包、精確傳輸問(wèn)題。分享給大家供大家參考,具體如下:
粘包:
-
發(fā)生原因:
當(dāng)調(diào)用send的時(shí)候,數(shù)據(jù)并不是即時(shí)發(fā)給客戶(hù)端的。而是放到了系統(tǒng)的socket發(fā)送緩沖區(qū)里,等緩沖區(qū)滿(mǎn)了、或者數(shù)據(jù)等待超時(shí)了,數(shù)據(jù)才會(huì)發(fā)送,所以有時(shí)候發(fā)送太快的話(huà),前一份數(shù)據(jù)還沒(méi)有傳給客戶(hù)端,那么這份數(shù)據(jù)和上一份數(shù)據(jù)一起發(fā)給客戶(hù)端的時(shí)候就會(huì)造成“粘包” 。
-
解決方案:
解決根源的思想是避免不同段的數(shù)據(jù)一起發(fā)送。
- 方案1:前一段數(shù)據(jù)send完后,等待一段時(shí)間再send第二段數(shù)據(jù)。缺點(diǎn):時(shí)間效率低,而且也無(wú)法完全避免問(wèn)題【因?yàn)椴磺宄撛O(shè)置多少時(shí)間才能保證前一份數(shù)據(jù)已經(jīng)發(fā)送】
- 方案2:握手機(jī)制:前一段數(shù)據(jù)send完后,嘗試recv,等待客戶(hù)端回應(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) #接收客戶(hù)端確認(rèn) print(ack) conn.send(b"second info") except ConnectionResetError as e: print(e) break server.close()
客戶(hù)端【接收方】代碼:
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()
不精確傳輸問(wèn)題:
發(fā)生原因:
由于數(shù)據(jù)太大,發(fā)送方一次send不完,而接收方只recv一次,使得影響了后面數(shù)據(jù)的傳輸
解決方案:
解決根源的思想是改變r(jià)ecv的次數(shù)。
- 方案:將數(shù)據(jù)的大小發(fā)給接收方,讓接收方來(lái)決定recv的次數(shù)
方案實(shí)現(xiàn)代碼【以解決長(zhǎng)數(shù)據(jù)shell命令傳輸為例】:
服務(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會(huì)導(dǎo)致異常,有些是沒(méi)有返回值的command if cmd_len==0: data="command has nothing return" cmd_len=len(data.encode()) ##因?yàn)檫@里前面沒(méi)有發(fā)送操作,所以不用擔(dān)心粘包,如果有則要考慮處理 conn.send(str(cmd_len).encode())#因?yàn)閘en結(jié)果是int,所以還要轉(zhuǎn)換 #這里要處理粘包 ack=conn.recv(1024) conn.send(data.encode()) except ConnectionResetError as e: print(e) break server.close()
客戶(hù)端【接收方】:
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()
- 利用這個(gè)原理可以實(shí)現(xiàn)文件傳輸,只要能確定接受次數(shù),就能保證文件傳輸?shù)拇笮≌_。
更多關(guān)于Python相關(guān)內(nèi)容可查看本站專(zhuān)題:《Python Socket編程技巧總結(jié)》、《Python數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Python函數(shù)使用技巧總結(jié)》、《Python字符串操作技巧匯總》、《Python入門(mén)與進(jìn)階經(jīng)典教程》及《Python文件與目錄操作技巧匯總》
希望本文所述對(duì)大家Python程序設(shè)計(jì)有所幫助。
相關(guān)文章
Python使用Joblib模塊實(shí)現(xiàn)加快任務(wù)處理速度
在Python編程中,處理大規(guī)模數(shù)據(jù)或者進(jìn)行復(fù)雜的計(jì)算任務(wù)時(shí),通常需要考慮如何提高程序的運(yùn)行效率,本文主要介紹了如何使用Joblib模塊來(lái)加快任務(wù)處理速度,需要的可以參考下2024-03-03python3使用pandas獲取股票數(shù)據(jù)的方法
今天小編就為大家分享一篇python3使用pandas獲取股票數(shù)據(jù)的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-12-12Python的Flask框架應(yīng)用調(diào)用Redis隊(duì)列數(shù)據(jù)的方法
這里為大家?guī)?lái)Python的Flask框架應(yīng)用調(diào)用Redis隊(duì)列數(shù)據(jù)的方法,從而能夠?qū)崿F(xiàn)異步無(wú)阻塞從而提高某些實(shí)時(shí)處理情況下程序的性能,需要的朋友可以參考下2016-06-06Python實(shí)現(xiàn)的NN神經(jīng)網(wǎng)絡(luò)算法完整示例
這篇文章主要介紹了Python實(shí)現(xiàn)的NN神經(jīng)網(wǎng)絡(luò)算法,結(jié)合完整實(shí)例形式分析了Python使用numpy、matplotlib及sklearn模塊實(shí)現(xiàn)NN神經(jīng)網(wǎng)絡(luò)相關(guān)算法實(shí)現(xiàn)技巧與操作注意事項(xiàng),需要的朋友可以參考下2018-06-06Python實(shí)現(xiàn)聊天機(jī)器人的示例代碼
這篇文章主要介紹了Python實(shí)現(xiàn)聊天機(jī)器人,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07