溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

如何解決Python socket連接中的粘包、精確傳輸問題

發(fā)布時間:2021-06-16 15:06:12 來源:億速云 閱讀:193 作者:小新 欄目:開發(fā)技術(shù)

小編給大家分享一下如何解決Python socket連接中的粘包、精確傳輸問題,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!

粘包:

  • 發(fā)生原因:

當(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. 方案1:前一段數(shù)據(jù)send完后,等待一段時間再send第二段數(shù)據(jù)。缺點:時間效率低,而且也無法完全避免問題【因為不清楚該設(shè)置多少時間才能保證前一份數(shù)據(jù)已經(jīng)發(fā)送】

    2. 方案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()

不精確傳輸問題:

發(fā)生原因:

由于數(shù)據(jù)太大,發(fā)送方一次send不完,而接收方只recv一次,使得影響了后面數(shù)據(jù)的傳輸

解決方案:

解決根源的思想是改變recv的次數(shù)。

  • 方案:將數(shù)據(jù)的大小發(fā)給接收方,讓接收方來決定recv的次數(shù)

方案實現(xiàn)代碼【以解決長數(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會導(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è)資訊頻道,感謝各位的閱讀!

向AI問一下細(xì)節(jié)

免責(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)容。

AI