服务器端代码

 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. C++this详解

    以前对this指针误解挺多的,在这里单独写一篇进行总结,有不对之处,欢迎指正批评! 一.问题 1.一个类中的不同对象在调用自己的成员函数时,其实它们调用的是同一段函数代码,那么成员函数如何知道要访问哪 ...

  2. [C++11新特性] 智能指针详解

    动态内存的使用很容易出问题,因为确保在正确的时间释放内存是极为困难的.有时我们会忘记释放内存产生内存泄漏,有时提前释放了内存,再使用指针去引用内存就会报错. 为了更容易(同时也更安全)地使用动态内存, ...

  3. [POI2008]激光发射器SZK

    Description 多边形相邻边垂直,边长为整数,边平行坐标轴.要在多边形的点上放一些激光发射器和接收器.满足下列要求: 1发射器和接收器不能放置在同一点: 2发射器发出激光可以沿壁反射,最终到达 ...

  4. 贪心/思维题 Codeforces Round #310 (Div. 2) C. Case of Matryoshkas

    题目传送门 /* 题意:套娃娃,可以套一个单独的娃娃,或者把最后面的娃娃取出,最后使得0-1-2-...-(n-1),问最少要几步 贪心/思维题:娃娃的状态:取出+套上(2),套上(1), 已套上(0 ...

  5. Kali linux 2016.2(Rolling)里的枚举服务

    前言 枚举是一类程序,它允许用户从一个网络中收集某一类的所有相关服务.

  6. [转]符号和运算符参考 (F#)

    本文转自:http://msdn.microsoft.com/zh-cn/library/dd233228.aspx 本主题包含一个表,其中列出了 F# 语言中使用的符号和运算符. 符号和运算符表   ...

  7. js的replace函数把"$"替换成成"\$"

    var aa = 18$    转换成   aa = 18\$ aa.replace("\$","\\\$");    注意JS的replace方法只能替换第一 ...

  8. SpringMVC -- 必知必会

    SpringMVC基于模型--视图--控制器(Model-View-Controller,MVC)模式实现,属于SpringFrameWork的后续产品,已经融合在SpringWebFlow里面.它通 ...

  9. 使用iconfont管理项目中的字体图标

    先来说说字体图标的好处: 很容易任意地缩放: 很容易地改变颜色: 很容易地产生阴影: 可以拥有透明效果: 一般来说,有先进的浏览器支持: 可以使用CSS来装饰(可以得到CSS很好支持): 可以快速转化 ...

  10. Android学习笔记(十五) Http

    1.Http协议概要 应用程序和服务间的请求/响应是无状态的,即响应完即断开连接. HttpClient库是Android自带的,故无需引入该库 2.Http请求和获取数据 生成代表客户端的HttpC ...