zoukankan      html  css  js  c++  java
  • 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:Users94269PycharmProjectsFlask2hellodd'
    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 是比较理想的状态

      

  • 相关阅读:
    【ARM-Linux开发】C语言getcwd()函数:取得当前的工作目录
    【ARM-Linux开发】C语言getcwd()函数:取得当前的工作目录
    【ARM-Linux开发】Gstreamer+QT+摄像头 编程总结
    【ARM-Linux开发】Gstreamer+QT+摄像头 编程总结
    【ARM-Linux开发】 pkg-config的用法
    【ARM-Linux开发】 pkg-config的用法
    【ARM-Linux开发】gstreamer教程及在DM3730上的应用
    【ARM-Linux开发】gstreamer教程及在DM3730上的应用
    【ARM-Linux开发】打包解包命令
    【ARM-Linux开发】打包解包命令
  • 原文地址:https://www.cnblogs.com/zhuangdd/p/12688334.html
Copyright © 2011-2022 走看看