# 粘包产生的原因
# 粘包问题主要还是因为接收方不知道消息之间的界限,不知道一次性提取多少字节的数据所造成的。
# 基于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