31 小文件上传下载、执行命令、粘包
一、文件上传下载
客户端发送一个字典:{上传下载指令,文件名,文件内容}到服务端 ,请求上传和下载
import socket import json sk = socket.socket() sk.bind(('127.0.0.1',8080)) sk.listen() conn, addr = sk.accept() str_dic = conn.recv(10240).decode('utf-8') dic = json.loads(str_dic) if dic['opt'] == 'upload': print('客户端选择的是上传功能') with open(dic['filename']+'(1)', 'w', encoding='utf-8') as f: f.write(dic['content']) elif dic['opt'] == 'download': print('客户端选择的是下载功能') # 收到客户端发来的请求字典 根据文件名来读取文件内容 with open(dic['filename'], 'r', encoding='utf-8') as f: content = f.read() dic['content'] = content dic_str = json.dumps(dic) conn.send(dic_str.encode('utf-8')) conn.close() sk.close()
import socket import json import os sk = socket.socket() sk.connect(('127.0.0.1',8080)) menu = {'1':'upload', '2':'download'} file_path = r'C:\Users\94269\PycharmProjects\Flask2\hello\dd' filename = os.path.basename(file_path) for k, v in menu.items(): print(k, v) num = input('请选择:') if num == '1': print('您选择的是上传功能') f = open(file_path, 'r', encoding='utf-8') content = f.read() f.close() dic = {'opt':menu.get(num), 'filename':filename, 'content':content} str_dic = json.dumps(dic) sk.send(str_dic.encode('utf-8')) if num == '2': print('您选择的是下载功能') # 发送字典:用户选择项,要下载的文件名,空的内容 dic_no = {'opt': menu.get(num), 'filename': filename, 'content': None} # 字典序列化 str_dic = json.dumps(dic_no) # 发送 sk.send(str_dic.encode('utf-8')) # 等待接收 服务端 发送的字典 dic_str = sk.recv(10240).decode('utf-8') # 字典反序列化 dic = json.loads(dic_str) # 防止文件覆盖 dic['filename'] = filename + '(副本)' # 写 文件 with open(dic['filename'], 'w', encoding='utf-8') as f: f.write(dic['content']) else: print('请输入正确的数字') sk.close()
(二)、执行命令
想到了 os模块的popen()方法
import os r = os.popen('dir') print(r.read())
可是,这个方法有一个缺点,写入错误的指令,不会报错,只是显示乱码
下面推荐一个新模块:
它能输出正确结果 也可以输出error
import subprocess r = subprocess.Popen('dir', shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE) # cmd : 代表系统命令 # shell = True 代表这条命令是 系统命令,告诉操作系统,将cmd当成系统命令去执行 # stdout 是执行完系统命令之后,用于保存结果的一个管道 # stderr 是执行完系统命令之后,用于保存错误结果的一个管道 print("stdout:", r.stdout.read().decode('gbk')) print("stderr:", r.stderr.read().decode('gbk'))
二、粘包
粘包
只有tcp协议才会发送粘包,udp不会发生
通俗的来讲 发送端发送数据,接收端不知道如何取去接收,造成的一种数据混乱的现象
在tcp协议中:
1、有一个合包机制(nagle算法),将多次连续发送且间隔较小的数据,进行打包成一块数据传送.
2、 还有一个机制是拆包机制,在发送端,因为受到网卡的MTU限制,会将大的超过MTU限制的 数据,进行拆分,拆分成多个小的数据,进行传输. 当传输到目标主机 的操作系统层时,会重新将多个小的数据合并成原本的数据
针对 使用udp协议发送数据,一次收发大小究竟多少合适?
udp不会发生粘包,udp协议本层对一次收发数据大小的限制是:
65535 - ip包头(20) - udp包头(8) = 65507
站在数据链路层,因为网卡的MTU一般被限制在了1500,所以对于数据链路层来说,一次收发数据的大小被限制在 1500 - ip包头(20) - udp包头(8) = 1472
得到结论:
如果sendto(num) num > 65507 报错
1472 < num < 65507 会在数据链路层拆包,而udp本身就是不可靠协议,所以一旦拆包之后,造成的多个小数据包在网络传输中,如果丢任何一个,那么此次数据传输失败
num < 1472 是比较理想的状态