一、解决粘包问题(low版)
问题的根源在于,接收端不知道发送端将要传送的字节流的长度,所以解决粘包的方法就是围绕,如何让发送端在发送数据前,把自己将要发送的字节流总大小让接收端知晓,然后接收端来一个死循环接收完所有数据。
1.1 服务端
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
|
import socket, subprocess
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind(('127.0.0.1', 8000)) server.listen(5)
while True: conn, addr = server.accept()
print('start...') while True: cmd = conn.recv(1024) print('cmd:', cmd)
obj = subprocess.Popen(cmd.decode('utf8'), shell=True, stderr=subprocess.PIPE, stdout=subprocess.PIPE)
stdout = obj.stdout.read()
if stdout: ret = stdout else: stderr = obj.stderr.read() ret = stderr
ret_len = len(ret)
conn.send(str(ret_len).encode('utf8'))
data = conn.recv(1024).decode('utf8')
if data == 'recv_ready': conn.sendall(ret)
conn.close()
server.close()
|
1.2 客户端
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
|
import socket
client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
client.connect(('127.0.0.1', 8000))
while True: msg = input('please enter your cmd you want>>>').strip()
if len(msg) == 0: continue
client.send(msg.encode('utf8')) length = int(client.recv(1024))
client.send('recv_ready'.encode('utf8'))
send_size = 0 recv_size = 0
data = b''
while recv_size < length: data = client.recv(1024) recv_size += len(data)
print(data.decode('utf8'))
|
1.3 为何low
程序的运行速度远快于网络传输速度,所以在发送一段字节前,先用send去发送该字节流长度,这种方式会放大网络延迟带来的性能损耗
二、补充struct模块
2.1 简单使用
![124-解决粘包问题-struct模块参数.png?x-oss-process=style/watermark]()
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
|
import struct import json
|
三、解决粘包问题
解决粘包问题的核心就是:为字节流加上自定义固定长度报头,报头中包含字节流长度,然后一次send到对端,对端在接收时,先从缓存中取出定长的报头,然后再取真实数据。
3.1 使用struct模块创建报头
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
|
import json import struct
header_dic = { 'filename': 'a.txt', 'total_size': 111111111111111111111111111111111222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222223131232, 'hash': 'asdf123123x123213x' }
header_json = json.dumps(header_dic)
header_bytes = header_json.encode('utf-8') print(len(header_bytes))
|
3.2 服务端
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
|
from socket import * import subprocess import struct import json
server = socket(AF_INET, SOCK_STREAM) server.bind(('127.0.0.1', 8000)) server.listen(5)
print('start...') while True: conn, client_addr = server.accept() print(conn, client_addr)
while True: cmd = conn.recv(1024)
obj = subprocess.Popen(cmd.decode('utf8'), shell=True, stderr=subprocess.PIPE, stdout=subprocess.PIPE)
stderr = obj.stderr.read() stdout = obj.stdout.read()
|
3.3 客户端
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37
|
from socket import * import json import struct
client = socket(AF_INET, SOCK_STREAM) client.connect(('127.0.0.1', 8000))
while True: cmd = input('please enter your cmd you want>>>')
if len(cmd) == 0: continue
client.send(cmd.encode('utf8'))
|
四、TCP协议粘包问题分析
1.nagle算法规定,TCP协议会将数据量较小、时间间隔短的数据合并为一条发送给客户端
4.1 服务端
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
|
from socket import *
server = socket(AF_INET, SOCK_STREAM) server.bind(('127.0.0.1', 8080)) server.listen(5)
conn, addr = server.accept()
|
4.2 客户端
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
|
Copyfrom socket import * import time
client = socket(AF_INET, SOCK_STREAM) client.connect(('127.0.0.1', 8080))
client.send(b'hello') client.send(b'world') client.send(b'lqz')
|
2.接收方不及时接收缓冲区的包,造成多个包接收(客户端发送了一段数据,服务端只收了一小部分,服务端下次再收的时候还是从缓冲区拿上次遗留的数据,产生粘包)
4.3 服务端
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
|
|
4.4 客户端