粘包问题发生的缘由:shell
1.发送端须要等缓冲区满才发送出去,形成粘包(发送数据时间间隔很短,数据了很小,会合到一块儿,产生粘包),这样接收端,就难于分辨出来了,必须提供科学的拆包机制。 即面向流的通讯是无消息保护边界的。json
2.接收方不及时接收缓冲区的包,形成多个包接收(客户端发送了一段数据,服务端只收了一小部分,服务端下次再收的时候仍是从缓冲区拿上次遗留的数据,产生粘包) 缓存
粘包问题主要仍是由于接收方不知道消息之间的界限,不知道一次性提取多少字节的数据所形成的。socket
一、把整型数字转成bytes类型 二、转成的bytes是固定长度的编码
import struct res=struct.pack('i',20332) # i:整型 print(res,len(res)) # b'lO\x00\x00' 4 res2=struct.unpack('i',res) print(res2[0]) # 20332
为字节流加上自定义固定长度报头,报头中包含字节流长度,而后一次send到对端,对端在接收时,先从缓存中取出定长的报头(报头中含有真实数据长度),而后再取真实数据spa
服务端:code
from socket import * import subprocess import struct ........ while True: conn,client_addr=server.accept() #(链接对象,客户端的ip和端口) print(client_addr) while True: try: cmd=conn.recv(1024) obj=subprocess.Popen(cmd.decode('utf-8'),# dir shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE ) stdout=obj.stdout.read() stderr=obj.stderr.read() total_size=len(stdout) + len(stderr)# 一、制做固定长度的报头 # 430 header=struct.pack('i',total_size) conn.send(header) # 二、发送报头 conn.send(stdout) #三、发送真实的数据 conn.send(stderr) #subprocess返回byte类型,但须要gbk解码 except ConnectionResetError: break conn.close() server.close()
客户端:server
from socket import * import struct .......... while True: cmd=input('>>>: ').strip() if not cmd:continue client.send(cmd.encode('utf-8')) # dir header=client.recv(4) #一、先收固定长度的报头 total_size=struct.unpack('i',header)[0] #二、解析报头 print(total_size) # 430 recv_size=0 #三、根据报头内的信息,收取真实的数据 res=b'' while recv_size < total_size: recv_data=client.recv(1024) res+=recv_data recv_size+=len(recv_data) print(res.decode('gbk')) client.close()
咱们能够把报头作成字典,字典里包含将要发送的真实数据的详细信息,字典而后json序列化,编码成byte类型,而后用struck将数据长度打包成4个字节(4个本身足够用了)对象
发送时:blog
先发报头长度,再编码报头内容而后发送,最后发真实内容
接收时:先收报头长度,用struct取出来,根据取出的长度收取报头内容,而后解码,反序列化,从反序列化的结果中取出待取数据的详细信息,而后去取真实的数据内容
服务端:
from socket import * import subprocess import struct import json ........... while True: conn,client_addr=server.accept() #(链接对象,客户端的ip和端口) print(client_addr) while True: try: cmd=conn.recv(1024) obj=subprocess.Popen(cmd.decode('utf-8'), shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE ) stdout=obj.stdout.read() stderr=obj.stderr.read() header_dic={ # 一、制做报头 'total_size':len(stdout) + len(stderr), 'md5':'123svsaef123sdfasdf', 'filename':'a.txt' } header_json = json.dumps(header_dic) header_bytes = header_json.encode('utf-8') header_size=len(header_bytes) # 二、先发送报头的长度 conn.send(struct.pack('i',header_size)) conn.send(header_bytes) # 三、发送报头 conn.send(stdout) # 四、发送真实的数据 conn.send(stderr) except ConnectionResetError: break conn.close() server.close()
客户端:
from socket import * import struct import json ......... while True: cmd=input('>>>: ').strip() if not cmd:continue client.send(cmd.encode('utf-8')) header_size=struct.unpack('i',client.recv(4))[0] #一、先收报头的长度 header_bytes=client.recv(header_size) #二、接收报头 header_json=header_bytes.decode('utf-8') #三、解析报头 header_dic=json.loads(header_json) print(header_dic) total_size=header_dic[ 'total_size'] #四、根据报头内的信息,收取真实的数据 # print(total_size) #1025 recv_size=0 res=b'' while recv_size < total_size: recv_data=client.recv(1024) res+=recv_data recv_size+=len(recv_data) print(res.decode('gbk')) client.close()
客户端实现等待后重连:
import socket import time while True: try: client = socket.socket(socket.AF_INET, socket.SOCK_STREAM) client.connect(('127.0.0.1',8080)) break except ConnectionRefusedError: time.sleep(3) print('等待3秒。。。')