Python3使用TCP编写一个简易的文件下载器功能


Posted in Python onMay 08, 2019

利用Python3来实现TCP协议,和UDP类似。UDP应用于及时通信,而TCP协议用来传送文件、命令等操作,因为这些数据不允许丢失,否则会造成文件错误或命令混乱。下面代码就是模拟客户端通过命令行操作服务器。客户端输入命令,服务器执行并且返回结果。

TCP(Transmission Control Protocol 传输控制协议):是一种面向连接的、可靠的、基于字节流的传输层通信协议,由IETF的RFC 793定义。

使用TCP编写一个简易的文件下载器要求:需编写文件下载器服务端和文件下载器客户端

服务器端代码:

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)服务器端

Python3使用TCP编写一个简易的文件下载器功能

2)客户端

Python3使用TCP编写一个简易的文件下载器功能

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

总结

以上所述是小编给大家介绍的Python3使用TCP编写一个简易的文件下载器功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

Python 相关文章推荐
Python多线程编程(六):可重入锁RLock
Apr 05 Python
Python实现配置文件备份的方法
Jul 30 Python
Python编程使用*解包和itertools.product()求笛卡尔积的方法
Dec 18 Python
Python编程二分法实现冒泡算法+快速排序代码示例
Jan 15 Python
Python使用paramiko操作linux的方法讲解
Feb 25 Python
python数据挖掘需要学的内容
Jun 23 Python
在Django admin中编辑ManyToManyField的实现方法
Aug 09 Python
Python写捕鱼达人的游戏实现
Mar 31 Python
Python退出时强制运行一段代码的实现方法
Apr 29 Python
Python实现CAN报文转换工具教程
May 05 Python
scrapy-redis分布式爬虫的搭建过程(理论篇)
Sep 29 Python
No module named ‘win32gui‘ 的解决方法(踩坑之旅)
Feb 18 Python
详解Python的三种可变参数
May 08 #Python
Python数据类型之Tuple元组实例详解
May 08 #Python
基于腾讯云服务器部署微信小程序后台服务(Python+Django)
May 08 #Python
python中正则表达式与模式匹配
May 07 #Python
详解Python3网络爬虫(二):利用urllib.urlopen向有道翻译发送数据获得翻译结果
May 07 #Python
python对象与json相互转换的方法
May 07 #Python
python使用threading.Condition交替打印两个字符
May 07 #Python
You might like
linux下安装php的memcached客户端
2014/08/03 PHP
PHP实现数组递归转义的方法
2014/08/28 PHP
PHP以json或xml格式返回请求数据的方法
2018/05/31 PHP
getElementsByTagName vs selectNodes效率 及兼容的selectNodes实现
2010/02/26 Javascript
JQuery EasyUI 对话框的使用方法
2010/10/24 Javascript
JavaScript中setInterval的用法总结
2013/11/20 Javascript
代码获取历史上的今天发生的事
2014/04/11 Javascript
JavaScript运动减速效果实例分析
2015/08/04 Javascript
easyui Draggable组件实现拖动效果
2015/08/19 Javascript
jQuery弹层插件jquery.fancybox.js用法实例
2016/01/22 Javascript
Bootstrap导航简单实现代码
2017/03/06 Javascript
js实现鼠标移动到图片产生遮罩效果
2017/10/21 Javascript
javaScript日期工具类DateUtils详解
2017/12/08 Javascript
解决vue页面DOM操作不生效的问题
2018/03/17 Javascript
Nodejs中的JWT和Session的使用
2018/08/21 NodeJs
JavaScript使用表单元素验证表单的示例代码
2019/08/20 Javascript
vue-cli3项目打包后自动化部署到服务器的方法
2020/09/16 Javascript
vue-cli —— 如何局部修改Element样式
2020/10/22 Javascript
[44:41]Fnatic vs Liquid 2018国际邀请赛小组赛BO2 第二场 8.16
2018/08/17 DOTA
Python 文件和输入输出小结
2013/10/09 Python
python计算对角线有理函数插值的方法
2015/05/07 Python
Python编写简单的HTML页面合并脚本
2016/07/11 Python
彻彻底底地理解Python中的编码问题
2018/10/15 Python
python字典一键多值实例代码分享
2019/06/14 Python
python 基于TCP协议的套接字编程详解
2019/06/29 Python
Python使用线程来接收串口数据的示例
2019/07/02 Python
tensorflow中tf.slice和tf.gather切片函数的使用
2020/01/19 Python
python3.7+selenium模拟淘宝登录功能的实现
2020/05/26 Python
Python3爬虫中Splash的知识总结
2020/07/10 Python
vue+django实现下载文件的示例
2021/03/24 Vue.js
2014信息技术专业毕业生自我评价
2014/01/17 职场文书
幼儿园秋游活动方案
2014/01/21 职场文书
个人廉洁自律承诺书
2014/03/27 职场文书
2015年新农村建设工作总结
2015/05/22 职场文书
ObjectMapper 如何忽略字段大小写
2021/06/29 Java/Android
nginx之内存池的实现
2022/06/28 Servers