PHP前端开发

详解python网络编程调用recv函数完整接收数据的三种方法

百变鹏仔 1个月前 (02-08) #Python
文章标签 三种

最近在使用python进行网络编程开发一个通用的tcpclient测试小工具。在使用socket进行网络编程中,如何判定对端发送一条报文是否接收完成,是进行socket网络开发必须要考虑的一个问题。这里,简要介绍一下判别数据接收接收完成常用的三种方法:

1.基础数据接收法:

 使用基础数据接收法时,当与服务socket断开连接时,会接收到空字符串。因此,可以根据此特点,在程序中加入循环,一直接收数据,直到数据发送端关闭socket连接。适用场景:客户端和服务器的链接为短链接(即一次socket通讯后连接就会关闭)。代码如下:

import socket,struct,sys,timePort=22220#assume a socket disconnect (data returned is empty string) means  all data was #done being sent.def recv_basic(the_socket):    total_data=[]    while True:        data = the_socket.recv(20480)            if not data: break        total_data.append(data)    return ''.join(total_data)

  2.尾标识方法

 通过寻找接收的协议数据中的尾标识字符串,获取完整的数据的数据报文。适用场景:接收的协议数据中包含相关的尾标识。代码如下:

立即学习“Python免费学习笔记(深入)”;

End='something useable as an end marker'def recv_end(the_socket):    total_data=[];data=''    while True:            data=the_socket.recv(8192)            if End in data:                total_data.append(data[:data.find(End)])                break            total_data.append(data)            if len(total_data)>1:                #check if end_of_data was split                last_pair=total_data[-2]+total_data[-1]                if End in last_pair:                    total_data[-2]=last_pair[:last_pair.find(End)]                    total_data.pop()                    break    return ''.join(total_data)

3.负载长度方法

     即通过协议数据中的负载长度值,确定有效报文的长度,适用场景:协议数据中包含负载协议字段。这种方法也是比较常用和通用的一种方法,但是需要一边接收数据一边解析数据。代码如下: 

def recv_size(the_socket):    #data length is packed into 4 bytes    total_len=0;total_data=[];size=sys.maxint    size_data=sock_data='';recv_size=8192    while total_len<size:>4:                size_data+=sock_data                size=struct.unpack('&gt;i', size_data[:4])[0]                recv_size=size                if recv_size&gt;524288:recv_size=524288                total_data.append(size_data[4:])            else:                size_data+=sock_data        else:            total_data.append(sock_data)        total_len=sum([len(i) for i in total_data ])    return ''.join(total_data)</size:>