tcp协议产生-粘包问题的解决方案

客户端

 1 客户端
 2 
 3 from socket import *
 4 import json,struct
 5 
 6 
 7 client=socket(AF_INET,SOCK_STREAM)
 8 client.connect(('127.0.0.1',8080))
 9 
10 
11 while True:
12     cmd=input('>>>').strip()
13     if len(cmd)==0:continue
14     client.send(cmd.encode('utf8'))
15     print('客户端命令发送完毕')
16     # 收到报头的长度
17     header_len=struct.unpack('i',client.recv(4))[0]
18 
19     # 收到报头的字节内容
20     header_bytes=client.recv(header_len)
21 
22     # 转成字符串
23     header_str=header_bytes.decode('utf8')
24 
25     # 根据json转成字典
26     header_dic=json.loads(header_str)
27 
28     # 取得文件总大小
29     total_size=header_dic['total_size']
30 
31     recv_size=0
32     
33     # 下面的 data 是字节,所以res也必须是字节
34     res=b''
35     while recv_size<total_size:
36         data=client.recv(1024)
37         res+=data
38         recv_size+=len(data)
39     print(res.decode('gbk'))
40 
41 client.close()

 

服务端

 1 服务端
 2 
 3 from socket import *
 4 import json,struct
 5 import subprocess
 6 
 7 server=socket(AF_INET,SOCK_STREAM)
 8 server.setsockopt(SOL_SOCKET,SO_REUSEADDR,1)
 9 server.bind(('127.0.0.1',8080))
10 server.listen()
11 
12 while True:
13     conn,client_addr=server.accept()
14     while True:
15         try:
16             cmd=conn.recv(1024)
17             if len(cmd)==0:break
18             obj=subprocess.Popen(cmd.decode('utf8'),
19                                  shell=True,
20                                  stdout=subprocess.PIPE,
21                                  stderr=subprocess.PIPE,
22                                  )
23             stdout=obj.stdout.read()
24             stderr=obj.stderr.read()
25             # 制作字典报头
26             header_dic={
27                 'file_name':'07 执行python程序的两种方式.mp4',
28                 'total_size':len(stderr)+len(stdout),
29                 'hash':'fkjds'
30             }
31             # 将报头用json 序列化
32             header_json=json.dumps(header_dic)
33             # 将json文件转成字节
34             header_bytes=header_json.encode('utf8')
35 
36             # 将报头打包发送给客户端
37             header=struct.pack('i',len(header_bytes))
38 
39             # 将4个字节的报头发送给客户端,客户端会收到压缩的4个字节的报头
40             conn.send(header)
41 
42             # 将字节的报头内容发送给客户端,因为上一步只是将报头的长度发送给了客户端
43             conn.send(header_bytes)
44 
45             # 发送正文
46             conn.send(stdout)
47             conn.send(stderr)
48         except ConnectionResetError:
49             break
50     conn.close()
51 
52 server.close()

 

posted @ 2018-07-09 22:06  Smart1san  阅读(197)  评论(0编辑  收藏  举报