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()
S
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()
C

 

 

(二)、执行命令

 想到了 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'))
subprocess 模块

 

二、粘包

   粘包

      只有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 是比较理想的状态

  

posted @ 2020-04-12 22:46  蜗牛般庄  阅读(143)  评论(0编辑  收藏  举报
Title
页脚 HTML 代码