服务器端代码

 import socket
 import os
 import threading

 # 处理客户端请求下载文件的操作(从主线程提出来的代码)
 def deal_client_request(ip_port, service_client_socket):
     # 连接成功后,输出“客户端连接成功”和客户端的ip和端口
     print("客户端连接成功", ip_port)
     # 接收客户端的请求信息
     file_name = service_client_socket.recv(1024)
     # 解码
     file_name_data = file_name.decode("utf-8")
     # 判断文件是否存在
     if os.path.exists(file_name_data):
         #输出文件字节数
         fsize = os.path.getsize(file_name_data)
         #转化为兆单位
         fmb = fsize/float(1024*1024)
         #要传输的文件信息
         senddata = "文件名:%s  文件大小:%.2fMB"%(file_name_data,fmb)
         #发送和打印文件信息
         service_client_socket.send(senddata.encode("utf-8"))
         print("请求文件名:%s  文件大小:%.2f MB"%(file_name_data,fmb))
         #接受客户是否需要下载
         options = service_client_socket.recv(1024)
         if options.decode("utf-8") == "y":
             # 打开文件
             with open(file_name_data, "rb") as f:
                 # 计算总数据包数目
                 nums = fsize/1024
                 # 当前传输的数据包数目
                 cnum = 0

                 while True:
                     file_data = f.read(1024)
                     cnum = cnum + 1
                     jindu = cnum/nums*100

                     print("当前已下载:%.2f%%"%jindu,end = "\r")
                     if file_data:
                         # 只要读取到数据,就向客户端进行发送
                         service_client_socket.send(file_data)
                     # 数据读完,退出循环
                     else:
                         print("请求的文件数据发送完成")
                         break
         else:
             print("下载取消!")
     else:
         print("下载的文件不存在!")
     # 关闭服务当前客户端的套接字
     service_client_socket.close()

 if __name__ == '__main__':
     # 把工作目录切换到data目录下
     os.chdir("./data")
     # 创建套接字
     tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
     # 绑定端口号
     tcp_server_socket.bind(("", 3356))
     # 设置监听,将主动套接字变为被动套接字
     tcp_server_socket.listen(128)

     # 循环调用accept,可以支持多个客户端同时连接,和多个客户端同时下载文件
     while True:
         service_client_socket, ip_port = tcp_server_socket.accept()
         # 连接成功后打印套接字号
         #print(id(service_client_socket))

         # 创建子线程
         sub_thread = threading.Thread(target=deal_client_request, args=(ip_port, service_client_socket))
         # 启动子线程
         sub_thread.start()

客户端代码

 # 多任务文件下载器客户端
 import socket

 if __name__ == '__main__':
     # 创建套接字
     tcp_client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
     # 和服务端连接
     server_ip = input("输入服务器IP:")
     tcp_client_socket.connect((server_ip, 3356))
     # 发送下载文件的请求
     file_name = input("请输入要下载的文件名:")
     # 编码
     file_name_data = file_name.encode("utf-8")
     # 发送文件下载请求数据
     tcp_client_socket.send(file_name_data)
     # 接收要下载的文件信息
     file_info = tcp_client_socket.recv(1024)
     # 文件信息解码
     info_decode = file_info.decode("utf-8")
     print(info_decode)
     #获取文件大小
     fileszie = float(info_decode.split(':')[2].split('MB')[0])
     fileszie2 = fileszie*1024
     # 是否下载?输入y 确认 输入q 取消
     opts = input("是否下载?(y 确认 q 取消)")
     if opts == 'q':
         print("下载取消!程序退出")
     else:
         print("正在下载 》》》")
         #向服务器确认正在下载
         tcp_client_socket.send(b'y')

         # 把数据写入到文件里
         with open("./" + file_name, "wb") as file:
             #目前接收到的数据包数目
             cnum = 0

             while True:
                 # 循环接收文件数据
                 file_data = tcp_client_socket.recv(1024)
                 # 接收到数据
                 if file_data:
                     # 写入数据
                     file.write(file_data)
                     cnum = cnum+1
                     jindu =cnum/fileszie2*100
                     print("当前已下载:%.2f%%"%jindu,end = "\r")
                 # 接收完成
                 else:
                     print("下载结束!")
                     break
     # 关闭套接字
     tcp_client_socket.close()

运行窗口

1)客户端

2)服务器端

注意:客户端和服务器端不要运行在idle中,直接终端运行

python实现tcp文件下载器的更多相关文章

  1. python sockerserver tcp 文件下载 udp

    #tcp serverclass MyHandler(socketserver.BaseRequestHandler): def handle(self): # 通信循环 while True: tr ...

  2. 使用网络TCP搭建一个简单文件下载器

    说明:该篇博客是博主一字一码编写的,实属不易,请尊重原创,谢谢大家! 目录 一丶项目介绍 二丶服务器Server 三丶测试TCP server服务器 四丶客户端Client 五丶测试客户端向服务器下载 ...

  3. tcp案例之文件下载器

    文件下载器客户端 import socket def main(): # 1.创建一个tcp socket tcp_client_socket=socket.socket(socket.AF_INET ...

  4. {每日一题}:tcp协议实现简单的文件下载器(单任务版)

    文件下载器客户端 这个版本的只是为了方便回顾一下TCP客服端,服务端的创建流程,缺点就是  服务器一次只能让一个人访问下载,过两个写个使用面向对象写一个多线程版的强化一下. from socket i ...

  5. python使用tcp实现一个简单的下载器

    上一篇中介绍了tcp的流程,本篇通过写一个简单的文件下载器程序来巩固之前学的知识. 文件下载器的流程如下: 客户端: 输入目标服务器的ip和port 输入要下载文件的名称 从服务器下载文件保存到本地 ...

  6. 【python】M3U8下载器脚本

    [python]M3U8下载器脚本 脚本目标: 1. 输入M3U8文件的链接,得到视频 2.使用异步操作,这样可以快很多,不加锁,因为懒得写,而且影响不大 已知条件: 1.m3u8文件其实就是一个记录 ...

  7. python高级之装饰器

    python高级之装饰器 本节内容 高阶函数 嵌套函数及闭包 装饰器 装饰器带参数 装饰器的嵌套 functools.wraps模块 递归函数被装饰 1.高阶函数 高阶函数的定义: 满足下面两个条件之 ...

  8. Unity3D 更新文件下载器

    使用说明: 1)远端更新服务器目录 Package |----list.txt |----a.bundle |----b.bundle 2)list.txt是更新列表文件 格式是 a.bundle|r ...

  9. [python基础]关于装饰器

    在面试的时候,被问到装饰器,在用的最多的时候就@classmethod ,@staticmethod,开口胡乱回答想这和C#的static public 关键字是不是一样的,等面试回来一看,哇,原来是 ...

随机推荐

  1. 状压入门--bzoj1087: [SCOI2005]互不侵犯King【状压dp】

    Description 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上 左下右上右下八个方向上附近的各一个格子,共8个格子. Input 只有一行, ...

  2. MySQL 一对多查询,合并多的一方的信息

    select   c.name,  (select   group_concat(name) from student   s   where  s.class_id =c.id ) from  cl ...

  3. IOS利用Core Text对文字进行排版 - 转

    原贴地址:http://hi.baidu.com/jwq359699768/blog/item/5df305c893413d0a7e3e6f7b.html core text 这个包默认是没有的,要自 ...

  4. Spring注解驱动开发之扩展原理

    前言:现今SpringBoot.SpringCloud技术非常火热,作为Spring之上的框架,他们大量使用到了Spring的一些底层注解.原理,比如@Conditional.@Import.@Ena ...

  5. 归并排序算法及其JS实现

    归并排序(MERGE-SORT)是利用归并的思想实现的排序方法,该算法采用经典的分治(divide-and-conquer)策略(分治法将问题分(divide)成一些小的问题然后递归求解,而治(con ...

  6. Redux中的异步操作

    异步操作的另一种方案就是让Action Creator返回一个Promise对象. 我们这边使用  redux-promise  中间件 import { createStore, applyMidd ...

  7. 【Python】列表解析表达式

    1.语法 [expression for iter_val in iterable] [expression for iter_val in iterable if cond_expr] 2.示例 s ...

  8. [翻译] API测试最佳实践 - 身份验证(Authentication)

    API测试最佳实践 - 身份验证 适用等级:高级 1. 概况 身份验证通常被定义为是对某个资源的身份的确认的活动,这里面资源的身份指代的是API的消费者(或者说是调用者).一旦一个用户的身份验证通过了 ...

  9. wget安装更新

    #查看当前wget版本信息 wget -V #下载 wget https://ftp.gnu.org/gnu/wget/wget-1.19.tar.gz #解压 tar xvf wget-1.19.t ...

  10. druid数据库连接池整合到SpringMvc

    1.maven项目加入相关的依赖 <dependency> <groupId>com.alibaba</groupId> <artifactId>dru ...