当前位置: 首页 > news >正文

python socket文件传输实现

简单版

server(服务端)

import socket
import subprocess
import struct
import json
import os

share_dir = r'E:\server\share'  # 需要传输的文件所在的文件夹

phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
phone.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)  # 回收重用端口10000
phone.bind(('127.0.0.1', 10000))  # 0-65535  0-1024给操作系统,
phone.listen(5)
print('stearting')
while True:  # 建链接循环
    conn, client_addr = phone.accept()
    print(client_addr)
    while True:  # 通信循环
        try:
            # 1.收命令
            res = conn.recv(1024)  # get jiaoyue.mp4
            # 2.解析命令,提取相应命令参数
            cmds = res.decode('utf-8').split()  # ['get', 'jiaoyue.mp4']
            filename = cmds[1]
            # 3.以读的方式打开

            # 1制作报头
            header_dic = {
                'filename': filename,
                'md5': 'xxdxx',
                'file_size': os.path.getsize(r'%s/%s' % (share_dir, filename)) #E:\study\第3模块,面向对象\网络编程\文件传输\server\share\jiaoyue.mp4
            }
            header_json = json.dumps(header_dic)
            header_bytes = header_json.encode('utf-8')

            # 2 发送报头长度
            conn.send(struct.pack('i', len(header_bytes)))  # 固定长度4

            # 3 发报头
            conn.send(header_bytes)
            # 4发真实数据
            with open('%s/%s' % (share_dir, filename), 'rb') as f:
                # conn.send(f.read())
                for a in f:
                    conn.send(a)
        except ConnectionResetError:
            break
    conn.close()
phone.close()

client(客户端)

import socket
import struct
import json

download_dir = r'E:\client\download'  # 文件存放地址

pc = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
pc.connect(('127.0.0.1', 10000))

while True:
    # 1.发命令
    cmd = input('>>>:').strip()  # get a.text
    if not cmd:
        continue
    pc.send(cmd.encode('utf-8'))

    # 2.接受文件内容,以写的方式打开一个新文件,写入客户端新文件中

    # 1收报头长度
    obj = pc.recv(4)
    header_size = struct.unpack('i', obj)[0]

    # 2接收报头
    header_bytes = pc.recv(header_size)

    # 3解析报头,对于数据的描述
    header_json = header_bytes.decode('utf-8')
    header_dic = json.loads(header_json)
    print(header_dic)
    total_size = header_dic['file_size']
    file_name = header_dic['filename']

    # 4 接受真实的数据
    with open('%s/%s' % (download_dir, file_name), 'wb') as f:
        recv_size = 0
        while recv_size < total_size:
            res = pc.recv(1024)
            f.write(res)
            recv_size += len(res)
            print('总大小:%s  已经下载大小:%s' % (total_size, recv_size))

pc.close()

 

优化之后的版本

server

import socket
import struct
import json
import os


share_dir = r'E:\server\share'  # 文件地址

def get(cmds, conn):
    filename = cmds[1]
    # 3.以读的方式打开,读取文件
    # 1制作报头
    header_dic = {
        'filename': filename,
        'md5': 'xxdxx',
        'file_size': os.path.getsize(r'%s/%s' % (share_dir, filename))  # E:\study\第3模块,面向对象\网络编程\文件传输\server\share\jiaoyue.mp4
    }
    header_json = json.dumps(header_dic)
    header_bytes = header_json.encode('utf-8')

    # 2 发送报头长度
    conn.send(struct.pack('i', len(header_bytes)))  # 固定长度4

    # 3 发报头
    conn.send(header_bytes)
    # 4发真实数据
    with open('%s/%s' % (share_dir, filename), 'rb') as f:
        # conn.send(f.read())
        for a in f:
            conn.send(a)

def put(pc):
    # 2.接受文件内容,以写的方式打开一个新文件,写入客户端新文件中
    # 1收报头长度
    obj = pc.recv(4)
    header_size = struct.unpack('i', obj)[0]

    # 2接收报头
    header_bytes = pc.recv(header_size)

    # 3解析报头,对于数据的描述
    header_json = header_bytes.decode('utf-8')
    header_dic = json.loads(header_json)
    print(header_dic)
    total_size = header_dic['file_size']
    file_name = header_dic['filename']

    # 4 接受真实的数据
    with open('%s/%s' % (share_dir, file_name), 'wb') as f:
        recv_size = 0
        while recv_size < total_size:
            res = pc.recv(1024)
            f.write(res)
            recv_size += len(res)
            print('总大小:%s  已经下载大小:%s' % (total_size, recv_size))

def run():
    phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    phone.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)  # 回收重用端口10000
    phone.bind(('127.0.0.1', 10000))  # 0-65535  0-1024给操作系统,
    phone.listen(5)
    while True:  # 建链接循环
        conn, client_addr = phone.accept()
        print(client_addr)
        while True:  # 通信循环
            try:
                # 1.收命令
                res = conn.recv(1024)  # get jiaoyue.mp4
                # 2.解析命令,提取相应命令参数
                cmds = res.decode('utf-8').split()  # ['get', 'jiaoyue.mp4']
                if cmds[0] == 'get':
                    get(cmds, conn)
                elif cmds[0] == 'put':
                    put(conn)
            except ConnectionResetError:
                break
        conn.close()
    phone.close()

if __name__ == '__main__':
    run()

client

import socket
import struct
import json
import os

download_dir = r'E:\client\download'  # 文件存放地址

def get(pc):
    # 2.接受文件内容,以写的方式打开一个新文件,写入客户端新文件中
    # 1收报头长度
    obj = pc.recv(4)
    header_size = struct.unpack('i', obj)[0]

    # 2接收报头
    header_bytes = pc.recv(header_size)

    # 3解析报头,对于数据的描述
    header_json = header_bytes.decode('utf-8')
    header_dic = json.loads(header_json)
    print(header_dic)
    total_size = header_dic['file_size']
    file_name = header_dic['filename']

    # 4 接受真实的数据
    with open('%s/%s' % (download_dir, file_name), 'wb') as f:
        recv_size = 0
        while recv_size < total_size:
            res = pc.recv(1024)
            f.write(res)
            recv_size += len(res)
            print('总大小:%s  已经下载大小:%s' % (total_size, recv_size))

def put(cmds, conn):
    filename = cmds[1]
    # 3.以读的方式打开,读取文件
    # 1制作报头
    header_dic = {
        'filename': filename,
        'md5': 'xxdxx',
        'file_size': os.path.getsize(r'%s/%s' % (download_dir, filename))
    }
    header_json = json.dumps(header_dic)
    header_bytes = header_json.encode('utf-8')

    # 2 发送报头长度
    conn.send(struct.pack('i', len(header_bytes)))  # 固定长度4

    # 3 发报头
    conn.send(header_bytes)
    # 4发真实数据
    send_size = 0
    with open('%s/%s' % (download_dir, filename), 'rb') as f:
        # conn.send(f.read())
        for a in f:
            conn.send(a)
            send_size += len(a)
            print(send_size)

def run():
    pc = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    pc.connect(('127.0.0.1', 10000))
    print(pc)
    while True:
        # 1.发命令
        inp = input('>>>:').strip()  # get a.text
        if not inp:
            continue
        pc.send(inp.encode('utf-8'))
        cmds = inp.split()
        if cmds[0] == 'get':
            get(pc)
        elif cmds[0] == 'put':
            put(cmds, pc)
    pc.close()

if __name__ == '__main__':
    run()

 

转载于:https://www.cnblogs.com/Xanderzyl/p/10735247.html

相关文章:

  • BZOJ3711 Druzyny 最值分治、线段树
  • jmeter5.1企业级应用功能详解
  • 面向对象的三大特性之封装
  • 数据结构:自定义数组队列
  • 异或的性质及运用
  • 20175215 2018-2019-2 第八周java课程学习总结
  • 我的java问题排查工具单
  • 记录一个pom文件
  • 2.4 hive创建表实例讲解
  • Cookie Session和自定义分页
  • SSM框架的优势?
  • 获得小黄衫有感
  • Hello2 Analysis
  • exe4j 使用记录(二):jar打包exe
  • ModBus-RTU详解
  • js数组之filter
  • nginx 负载服务器优化
  • React Transition Group -- Transition 组件
  • Unix命令
  • ViewService——一种保证客户端与服务端同步的方法
  • 实战|智能家居行业移动应用性能分析
  • 小程序button引导用户授权
  • Unity3D - 异步加载游戏场景与异步加载游戏资源进度条 ...
  • 测评:对于写作的人来说,Markdown是你最好的朋友 ...
  • ​LeetCode解法汇总2696. 删除子串后的字符串最小长度
  • ![CDATA[ ]] 是什么东东
  • #include<初见C语言之指针(5)>
  • #基础#使用Jupyter进行Notebook的转换 .ipynb文件导出为.md文件
  • (附源码)spring boot基于小程序酒店疫情系统 毕业设计 091931
  • (附源码)流浪动物保护平台的设计与实现 毕业设计 161154
  • (没学懂,待填坑)【动态规划】数位动态规划
  • (十)DDRC架构组成、效率Efficiency及功能实现
  • (四)docker:为mysql和java jar运行环境创建同一网络,容器互联
  • (转) Android中ViewStub组件使用
  • (转)为C# Windows服务添加安装程序
  • .NET : 在VS2008中计算代码度量值
  • .net CHARTING图表控件下载地址
  • .Net中ListT 泛型转成DataTable、DataSet
  • .Net转前端开发-启航篇,如何定制博客园主题
  • :=
  • [ANT] 项目中应用ANT
  • [Big Data - Kafka] kafka学习笔记:知识点整理
  • [BZOJ 2142]礼物(扩展Lucas定理)
  • [CC-FNCS]Chef and Churu
  • [CQOI 2011]动态逆序对
  • [delphi]保证程序只运行一个实例
  • [flume$2]记录一个写自定义Flume拦截器遇到的错误
  • [HDOJ4911]Inversion
  • [ios-必看] IOS调试技巧:当程序崩溃的时候怎么办 iphone IOS
  • [Java] 图说 注解
  • [JDBC-1] JDBC Base Template
  • [JS]JavaScript 注释 输入输出语句
  • [Linux] - 定时任务crontab
  • [Luogu 3958] NOIP2017 D2T1 奶酪
  • [Node + Docker] 聊聊怎么把 nodeclub 构建成 Docker 镜像