# 粘包产生的原因
# 粘包问题主要还是因为接收方不知道消息之间的界限,不知道一次性提取多少字节的数据所造成的。
# 基于tcp协议的套接字会有粘包现象,而基于udp协议的套接字不会产生粘包现象
# tcp是基于数据流的,于是收发的消息不能为空,这就需要在客户端和服务端都添加空消息的处理机制,防止程序卡住;而udp是基于数据报的,即使你输入的是空内容,那也不是空消息,udp协议会帮你封装上消息头(ip+端口的方式),这样就有了消息办界
# 两种情况下会发生粘包
# 1、发送端需要等缓冲区满才发送出去,造成粘包(发送数据时间间隔很短,数据很小,就会合到一起,产生粘包)
# 2、接收方不及时接收缓冲区的包,造成多个包接收(客户端发送了一段数据 ,服务端只收了一小部分,服务端下次再收的时候还是从缓冲区拿上次遗留的数据,产生粘包)
server.py
# socket 基于tcp实现远程执行命令(解决粘包)low
# 此为low版,因为服务端与客户端会多发送接收一个数据包,会影响性能
from socket import *
import subprocess
ip_port = ('127.0.0.1', 8080)
back_log = 5
buffer_size = 1024
tcp_server = socket(AF_INET, SOCK_STREAM)
tcp_server.bind(ip_port)
tcp_server.listen(back_log)
while True:
conn, addr = tcp_server.accept()
while True:
try:
cmd = conn.recv(buffer_size)
if not cmd: break
cmd = cmd.decode('utf-8')
print('收到客户端命令', cmd)
res = subprocess.Popen(cmd, shell=True, # 第一个参数:命令字符串,第二个参数指定由shell处理
stderr=subprocess.PIPE, # 将基本的输入、输出及错误都放入管道
stdin=subprocess.PIPE, # 这些在管道里的信息都是字节形式,编码为utf-8
stdout=subprocess.PIPE
)
err = res.stderr.read() # 定义一个err变量接收基本的错误信息
if err: # 如果错误信息不为空
cmd_res = err # 输出的结果为基本的错误信息
else:
cmd_res = res.stdout.read() # 输出的结果为基本的输出信息
if not cmd_res: # 有些命令无返回结果,需要进行判断
cmd_res = '该命令没有返回结果'.encode('gbk')
# 解决粘包
length = len(cmd_res) # 获取发送执行结果的长度
conn.send(str(length).encode('utf-8')) # 向客户端发送长度,这里注意要发str类型
client_ready = conn.recv(buffer_size) # 收取客户端发来的ready信息,这样做是避免服务端直接两次发包又造成粘包
if client_ready == b'ready': # 如果服务端收到ready,则向客户端发送执行的结果
conn.send(cmd_res) # 向客户端发送执行的结果
except Exception:
break
conn.close()
tcp_server.close()
client.py
from socket import *
ip_port = ('127.0.0.1', 8080)
buff_size = 1024
tcp_client = socket(AF_INET, SOCK_STREAM)
tcp_client.connect(ip_port)
while True:
cmd = input('请输入命令').strip()
if not cmd: continue
if cmd == 'quit': break
cmd = cmd.encode('utf-8')
tcp_client.send(cmd)
# 解决粘包
length = tcp_client.recv(buff_size) # 客户端先接收服务端发来的长度
tcp_client.send(b'ready') # 客户端收到长度后再向服务端发包,让服务端接收,避免服务端直接两次发包又造成粘包
length = int(length.decode('utf-8')) # 将客户端发来的字符串形式的长度解码后转为数字类型
# cmd_res = tcp_client.recv(length) # 这里不能这样写,如果服务端发来的信息过大,是无法全部放入客户端的缓冲区中,应当用循环依次取出
recv_size = 0 # 设置收取的长度初始值为0
recv_msg = b'' # 设置收取的信息初始值为二进制空
while recv_size < length: # 循环收取服务端发来的信息
recv_msg += tcp_client.recv(buff_size) # 每次循环将获的数据都放入recv_msg
recv_size = len(recv_msg) # 每次循环获得实际收取的长度
print('命令执行的结果是', recv_msg.decode('gbk')) # windows系统默认编码为gbk