Python socket连接中的粘包、精确传输问题实例分析

作者:随风行云 时间:2023-12-21 23:42:48 

本文实例讲述了Python socket连接中的粘包、精确传输问题。分享给大家供大家参考,具体如下:

粘包:

  • 发生原因:

当调用send的时候,数据并不是即时发给客户端的。而是放到了系统的socket发送缓冲区里,等缓冲区满了、或者数据等待超时了,数据才会发送,所以有时候发送太快的话,前一份数据还没有传给客户端,那么这份数据和上一份数据一起发给客户端的时候就会造成“粘包” 。

  • 解决方案:

解决根源的思想是避免不同段的数据一起发送。

    1. 方案1:前一段数据send完后,等待一段时间再send第二段数据。缺点:时间效率低,而且也无法完全避免问题【因为不清楚该设置多少时间才能保证前一份数据已经发送】

    2. 方案2:握手机制:前一段数据send完后,尝试recv,等待客户端回应,确认第一段数据发送完后,再send第二段数据。完美方案?

方案二的演示:

服务端【发送方】代码:


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) #接收客户端确认
     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")#发送确认
data=client.recv(1024)
print(data.decode())
client.close()

不精确传输问题:

发生原因:

由于数据太大,发送方一次send不完,而接收方只recv一次,使得影响了后面数据的传输

解决方案:

解决根源的思想是改变recv的次数。

  • 方案:将数据的大小发给接收方,让接收方来决定recv的次数

方案实现代码【以解决长数据shell命令传输为例】:

服务端【发送方】:


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)
     #发现这里如果cmd_len为0会导致异常,有些是没有返回值的command
     if cmd_len==0:
       data="command has nothing return"
       cmd_len=len(data.encode())
     ##因为这里前面没有发送操作,所以不用担心粘包,如果有则要考虑处理
     conn.send(str(cmd_len).encode())#因为len结果是int,所以还要转换
     #这里要处理粘包
     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()
  • 利用这个原理可以实现文件传输,只要能确定接受次数,就能保证文件传输的大小正确。

希望本文所述对大家Python程序设计有所帮助。

来源:https://www.cnblogs.com/progor/p/8429893.html

标签:Python,socket连接
0
投稿

猜你喜欢

  • 详解python多线程、锁、event事件机制的简单使用

    2022-03-16 19:48:25
  • Python之os模块案例详解

    2021-06-18 15:09:38
  • 如何用表单在线建立目录?

    2010-06-16 09:49:00
  • Python迭代和迭代器详解

    2023-11-20 08:52:28
  • python爬取m3u8连接的视频

    2023-06-18 13:40:21
  • MybatisPlus如何处理Mysql的json类型

    2024-01-27 14:23:19
  • matplotlib.pyplot绘图显示控制方法

    2023-07-03 15:44:20
  • 详解Python中的from..import绝对导入语句

    2022-08-10 10:01:33
  • python Multiprocessing.Pool进程池模块详解

    2023-08-25 09:24:59
  • python批量翻译excel表格中的英文

    2022-11-16 08:09:07
  • Keras 在fit_generator训练方式中加入图像random_crop操作

    2023-10-31 23:05:51
  • python wav模块获取采样率 采样点声道量化位数(实例代码)

    2023-04-26 23:25:28
  • centos7 PHP环境搭建 GD库 等插件安装方法

    2023-11-05 21:25:38
  • mysql中#{}和${}的区别详解

    2024-01-12 21:37:17
  • 请谨慎对待程序的图标和名称

    2011-06-16 20:35:22
  • Python Django2.0集成Celery4.1教程

    2023-10-17 21:28:52
  • Python 利用Entrez库筛选下载PubMed文献摘要的示例

    2021-05-27 11:35:01
  • 详解Pytest测试用例的执行方法

    2022-02-15 18:28:14
  • Javascript将string类型转换int类型

    2023-09-17 00:05:37
  • javascript 通过键名获取键盘的keyCode方法

    2024-04-22 22:18:21
  • asp之家 网络编程 m.aspxhome.com