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

优化基于tcp,socket的ftp文件传输程序

原始程序:

template_ftp_server_old.py:

import socket
import json
import struct
import os
import time
import pymysql.cursorssoc = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
HOST = '192.168.31.111'
PORT = 4101
soc.bind((HOST,PORT))
port_str = str(PORT)
slice_id = int(port_str[:2]) - 20
ueX_str = 'user'
ue_id = int(port_str[3])
ueX_delay = ueX_str + str(ue_id) + '_delay' # 
ueX_v = ueX_str + str(ue_id) + '_v' #
ueX_loss = ueX_str + str(ue_id) + '_loss' #
soc.listen(5)
cnx = pymysql.connect(host='localhost', port=3306, user='root', password='123456', db='qt', charset='utf8mb4', connect_timeout=20)
cursor = cnx.cursor()# 上传函数
def uploading_file():ftp_dir = r'C:/Users/yinzhao/Desktop/ftp/ftpseverfile'  # 将传送的name.txt保存在该文件夹当中if not os.path.isdir(ftp_dir):os.mkdir(ftp_dir)head_bytes_len = conn.recv(4)  # 拿到struct后的头长度head_len = struct.unpack('i', head_bytes_len)[0]  # 取出真正的头长度# 拿到真正的头部内容conn.send("1".encode('utf8'))head_bytes = conn.recv(head_len)conn.send("1".encode('utf8'))# 反序列化后取出头head = json.loads(head_bytes)file_name = head['file_name']print(file_name)file_path = os.path.join(ftp_dir, file_name)data_len = head['data_len']received_size = 0data=b''#****************************************************旧程序问题1:一次发送最大数据量,越大传输速率越快*************************************max_size=65536*4while received_size < data_len:#************************************************旧程序问题2:在接收文件的循环中有大量统计 计算 操作数据库的代码,占用时间**************8#    ftp_v = open('E:\FTP_test\FTP_v.txt', 'w', encoding='utf-8')#  ftp_delay = open('E:\FTP_test\FTP_delay.txt', 'w', encoding='utf-8')if data_len - received_size > max_size:size = max_sizeelse:size = data_len - received_sizereceive_data = conn.recv(size)  # 接收文本内容#************************************************旧程序问题3:基于socket的tcp无需手动在服务端进行ack的操作,既若无需要服务端可以不手动回发消息**********# conn.send("1".encode('utf8'))#      delay=conn.recv(1024).decode('utf8')  # 接受时延#     conn.send(str(size).encode('utf-8'))  ##      v = conn.recv(1024).decode('utf8')  # 接受速率#      conn.send("1".encode('utf-8'))  ##      loss = conn.recv(1024).decode('utf8')  # 接受loss#      conn.send("1".encode('utf-8'))# print("delay:",delay)#      delay_float = float(delay)#     v_float = float(v)#     loss_float = float(loss)#     formatted_delay = "{:.2f}".format(delay_float)#      formatted_v = "{:.2f}".format(v_float)#    formatted_loss = "{:.2f}".format(loss_float)#    cursor.execute("UPDATE slice_v2 SET " + ueX_v +  "= %s," + ueX_delay +  "=%s, " + ueX_loss +  " =%s WHERE slice_id = %s", [str(formatted_v), str(formatted_delay), str(formatted_loss), slice_id])#    cnx.commit()data+=receive_datareceive_datalen = len(receive_data)received_size += receive_datalen# print(delay, file=ftp_delay)# print(v, file=ftp_v)#print(loss)# ftp_delay.close()#ftp_v.close()#***********************************************旧程序问题4:接收文件后进行组装,组装后统一写入,且在一个线程里,耗时*******************************with open(file_path, 'wb') as fw:fw.write(data)#  print("上传文件 " + str(file_name) + " 成功")#uploading_file()while True:encoding = 'utf-8'print('等待客户端连接...')conn, addr = soc.accept()print('客户端已连接:', addr)uploading_file()conn.close()

client_renew_old.py

# coding: UTF-8
import socket
import os
import struct
import json
import hashlib
import time# lient_soc.connect(('192.168.1.101', 8025))# 上传函数
def uploading_file(filepath,filename):client_soc = socket.socket()client_soc.connect(('192.168.31.111', 4101))try:encoding = 'utf-8'with open(filepath+filename, 'rb') as fr:#发送固定文件data = fr.read()head = {'data_len': len(data), 'file_name': filename}  # 自定义头head_bytes = json.dumps(head).encode('utf8')head_bytes_len = struct.pack('i', len(head_bytes))client_soc.send(head_bytes_len)client_soc.recv(20).decode('utf8')client_soc.send(head_bytes)client_soc.recv(20).decode('utf8')print(str(int(len(data))/1024)+'KB')# client_soc.send(data)max_size = 65536*4#max_size = 1024# 计算需要分成几个数据包发送num_packets = len(data) // max_sizeif len(data) % max_size > 0:num_packets += 1# 循环分批发送数据for i in range(num_packets):# 将延时、速率传到txt文件中start = i * max_sizeend = start + max_size# if(i==num_packets-1):#     end=len(data)packet = data[start:end]#从这里开始计时start = time.perf_counter()client_soc.sendall(packet)# client_soc.recv(1024).decode('utf8')#计算时间段# delay=float(time.perf_counter() - start)# s#v=round(len(packet)/1048576/delay,2)#MB/S# delay=int(delay*1000)#  print("延时:",delay)# print("传输速率:",v)#client_soc.send(str(delay).encode('utf8'))#      rev_size=client_soc.recv(1024).decode('utf8')#      loss=(len(packet)-int(rev_size))/len(packet)#   client_soc.send(str(v).encode('utf8'))#     client_soc.recv(1024).decode('utf8')#    client_soc.send(str(loss).encode('utf8'))#      client_soc.recv(1024).decode('utf8')#  print("丢包率:",loss)print("发送完毕")#uploading_file()except socket.error as e:print("发生错误:", str(e))finally:# 关闭连接client_soc.close()if __name__ == '__main__':filepath='C:/Users/yinzhao/Desktop/ftp/'filename='test.7z'i = 0while i<2 :encoding = "utf8"uploading_file(filepath,filename)i = i+1print(i)

目前有如下支持:

  1. 传输任何文件,若成功传输到服务器,不会损坏
  2. 传输速率:
    max_size =655364
    在这里插入图片描述
    max_size =65536
    8
    在这里插入图片描述

该代码经过测试,有如下问题与特性:
4. 一次发送最大数据量,越大传输速率越快。之前每次发送1024,现在修改为65536*4
5. 在接收文件的循环中有大量统计 计算 操作数据库的代码,占用时间
6. 基于socket的tcp无需手动在服务端进行ack的操作,既若无需要服务端可以不手动回发消息。原程序有大量手动进行确认的过程,占用大量时间
7. 完整接收文件后再进行组装,组装后统一写入,也可以改成接收一次包写入一次
8. 所有程序都在同一个线程里,可以将数据传输 统计 数据写入放在不同的程序里,提供程序执行速度

修改代码

import socket
import json
import struct
import os
import time
import pymysql.cursors
import datetime
import threadingsoc = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
HOST = '192.168.84.1'
PORT = 4101
soc.bind((HOST,PORT))
port_str = str(PORT)
slice_id = int(port_str[:2]) - 20
ueX_str = 'user'
ue_id = int(port_str[3])
ueX_delay = ueX_str + str(ue_id) + '_delay' # 
ueX_v = ueX_str + str(ue_id) + '_v' #
ueX_loss = ueX_str + str(ue_id) + '_loss' #
soc.listen(5)
cnx = pymysql.connect(host='localhost', port=3306, user='root', password='123456', db='qt', charset='utf8mb4', connect_timeout=20)
cursor = cnx.cursor()def up_to_mysql(v,delay,loss):delay_float = float(delay)v_float = float(v)loss_float = float(loss)formatted_delay = "{:.2f}".format(delay_float)formatted_v = "{:.2f}".format(v_float)formatted_loss = "{:.2f}".format(loss_float)cursor.execute("UPDATE slice_v2 SET " + ueX_v +  "= %s," + ueX_delay +  "=%s, " + ueX_loss +  " =%s WHERE slice_id = %s", [str(formatted_v), str(formatted_delay), str(formatted_loss), slice_id])cnx.commit()def write_file(file_path,data):with open(file_path, 'wb') as fw:fw.write(data)
# 上传函数
def uploading_file():ftp_dir = r'/home/lab/changeyaml/test'  # 将传送的name.txt保存在该文件夹当中if not os.path.isdir(ftp_dir):os.mkdir(ftp_dir)head_bytes_len = conn.recv(4)  # 拿到struct后的头长度head_len = struct.unpack('i', head_bytes_len)[0]  # 取出真正的头长度# 拿到真正的头部内容conn.send("1".encode('utf8'))head_bytes = conn.recv(head_len)conn.send("1".encode('utf8'))# 反序列化后取出头head = json.loads(head_bytes)file_name = head['file_name']print(file_name)file_path = os.path.join(ftp_dir, file_name)data_len = head['data_len']received_size = 0data=b''max_size=65536*8time_begin = datetime.datetime.now()time_rev_begin=datetime.datetime.now()time_rev_end=datetime.datetime.now()datasize_rev_last=0while received_size < data_len:if data_len - received_size > max_size:size = max_sizeelse:size = data_len - received_sizereceive_data = conn.recv(size)  # 接收文本内容  data+=receive_datareceive_datalen = len(receive_data)received_size += receive_datalentime_rev_end=datetime.datetime.now()time_gap=time_rev_end-time_rev_beginif time_gap.total_seconds()>=1 :time_rev_begin=time_rev_end# print((time_rev_end-time_rev_begin).total_seconds)data_rev_1s_KB=(received_size-datasize_rev_last)/1024# print("data:"+str(data_rev_1s_KB)+'KB')# print("speed:"+str(data_rev_1s_KB)+'KB/s')# print("speed:"+str(data_rev_1s_KB/1024)+'MB/s')delay= 1/(data_rev_1s_KB*1024/max_size)# print("delay:"+str(delay)+'s')delay_ms=delay*1000print("delay:"+str(delay)+'ms')datasize_rev_last=received_sizeup_mysql_thread = threading.Thread(target=up_to_mysql, args=(8*data_rev_1s_KB/1024, delay_ms,0))up_mysql_thread.start()time_end= datetime.datetime.now()time_transport=time_end-time_begintime_transport_s=time_transport.total_seconds()print("耗时:"+str(time_transport_s))data_len_KB=data_len/1024data_len_MB=data_len/(1024*1024)print("数据量:"+str(data_len_KB)+"KB")v_average=data_len_KB/time_transport_s# print("平均速度:"+str(v_average)+'KB/s')print("平均速度:"+str(data_len_MB/time_transport_s)+'MB/s')write_file_thread = threading.Thread(target=write_file, args=(file_path, data))write_file_thread.start()while True:encoding = 'utf-8'print('等待客户端连接...')conn, addr = soc.accept()print('客户端已连接:', addr)uploading_file()conn.close()
# coding: UTF-8
import socket
import os
import struct
import json
import hashlib
import time# 上传函数
def uploading_file(filepath,filename):client_soc = socket.socket()client_soc.connect(('192.168.1.21', 4101))try:encoding = 'utf-8'with open(filepath+filename, 'rb') as fr:#发送固定文件data = fr.read()head = {'data_len': len(data), 'file_name': filename}  # 自定义头head_bytes = json.dumps(head).encode('utf8')head_bytes_len = struct.pack('i', len(head_bytes))client_soc.send(head_bytes_len)client_soc.recv(20).decode('utf8')client_soc.send(head_bytes)client_soc.recv(20).decode('utf8')print(str(int(len(data))/1024)+'KB')# client_soc.send(data)max_size = 65536*8#max_size = 1024# 计算需要分成几个数据包发送num_packets = len(data) // max_sizeif len(data) % max_size > 0:num_packets += 1# 循环分批发送数据for i in range(num_packets):start = i * max_sizeend = start + max_size# if(i==num_packets-1):#     end=len(data)packet = data[start:end]#从这里开始计时start = time.perf_counter()client_soc.sendall(packet)print("发送完毕")#uploading_file()except socket.error as e:print("发生错误:", str(e))finally:# 关闭连接client_soc.close()if __name__ == '__main__':filepath='C:/Users/QD689/Desktop/ftp1.01/'filename='name.txt'i = 0while i<3 :encoding = "utf8"uploading_file(filepath,filename)uploading_file(filepath,filename)i = i+1print(i)

经过精简代码与多线程使用,以及合理配置每次发送数据量大小,ftp文件传输速率极大提高了,并且可以正确测算到速率与时延

相关文章:

优化基于tcp,socket的ftp文件传输程序

原始程序&#xff1a; template_ftp_server_old.py&#xff1a; import socket import json import struct import os import time import pymysql.cursorssoc socket.socket(socket.AF_INET, socket.SOCK_STREAM) HOST 192.168.31.111 PORT 4101 soc.bind((HOST,PORT)) p…...

MySQL 数据库 【增删查改(二)】

目录 一、表的设计 1、一对一 2、一对多 3、多对多 二、新增 三、查询 1、聚合查询 &#xff08;1&#xff09;聚合函数&#xff1a; &#xff08;2&#xff09; group by 子句 &#xff08;3&#xff09;having 2、联合查询 (1)内连接 (2)外连接 (3)自链接 (4)…...

力扣 -- 978. 最长湍流子数组

一、题目 二、解题步骤 下面是用动态规划的思想解决这道题的过程&#xff0c;相信各位小伙伴都能看懂并且掌握这道经典的动规题目滴。 三、参考代码 class Solution { public:int maxTurbulenceSize(vector<int>& nums) {int nnums.size();vector<int> f(n);…...

甘特图 Dhtmlx Gantt

介绍 在一些任务计划、日程进度等场景中我们会使用到甘特图&#xff0c;Dhtmlx Gantt 对于甘特图的实现支持很友好&#xff0c;文档API介绍全面&#xff0c;虽然增强版的收费&#xff0c;但免费版的足以够用。 官网&#xff1a;https://docs.dhtmlx.com/gantt/ 安装dhtml gannt…...

iOS 应用上架流程详解

iOS 应用上架流程详解 欢迎来到我的博客&#xff0c;今天我将为大家分享 iOS 应用上架的详细流程。在这个数字化时代&#xff0c;移动应用已经成为了人们生活中不可或缺的一部分&#xff0c;而 iOS 平台的 App Store 则是开发者们发布应用的主要渠道之一。因此&#xff0c;了解…...

Python入门【LEGB规则、面向对象简介、面向过程和面向对象思想、面向对象是什么? 对象的进化 、类的定义、对象完整内存结构 】(十三)

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱敲代码的小王&#xff0c;CSDN博客博主,Python小白 &#x1f4d5;系列专栏&#xff1a;python入门到实战、Python爬虫开发、Python办公自动化、Python数据分析、Python前后端开发 &#x1f4e7;如果文章知识点有错误…...

【消息中间件】原生PHP对接Uni H5、APP、微信小程序实时通讯消息服务

文章目录 视频演示效果前言一、分析二、全局注入MQTT连接1.引入库2.写入全局连接代码 二、PHP环境建立总结 视频演示效果 【uniapp】实现买定离手小游戏 前言 Mqtt不同环境问题太多&#xff0c;新手可以看下 《【MQTT】Esp32数据上传采集&#xff1a;最新mqtt插件&#xff08;支…...

【C语言初阶】指针篇—上

目录 1. 指针是什么&#xff1f;2. 指针和指针类型2.1 指针-整数2.2 指针的解引用 3. 野指针3.1 野指针成因1. 指针未初始化2. 指针越界访问3. 指针指向的空间释放 3.2 如何规避野指针 1. 指针是什么&#xff1f; 指针是什么&#xff1f; 指针理解的2个要点&#xff1a; > 1…...

基于FasterRCNN深度学习网络的车辆检测算法matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 MATLAB2022A 3.部分核心程序 ....................................................................... % 训练Faster R-…...

机器学习深度学习——多层感知机

&#x1f468;‍&#x1f393;作者简介&#xff1a;一位即将上大四&#xff0c;正专攻机器学习的保研er &#x1f30c;上期文章&#xff1a;机器学习&&深度学习——感知机 &#x1f4da;订阅专栏&#xff1a;机器学习&&深度学习 希望文章对你们有所帮助 上一节…...

Django模型将模型注释同步到数据库

1、安装django-comment-migrate库 pip install django-comment-migrate 2、将库注册到settings.py文件中 INSTALLED_APPS [...django_comment_migrate, # 表注释... ] 3、加注释 3.1、给模型&#xff08;表&#xff09;加注释 在模型的class Meta中编辑 verbose_name&…...

STM32 Flash学习(二)

STM32F1的官方固件库操作FLASH的几个常用函数。这些函数和定义分布在源文件stm32f1xx_hal_flash.c/stm32f1xx_hal_flash_ex.c以及头文件stm32f1xx_hal_flash.h/stm32f1xx_hal_flash_ex.h中。 锁定解函数 对FLASH进行写操作前必须先解锁&#xff0c;解锁操作&#xff1a;在FLA…...

kotlin获取泛型集合的类型信息

通过 reified 关键字和内联函数来实现 inline fun <reified T> getClassFromList(list: List<T>): Class<T> {return T::class.java }fun main() {val list listOf("Hello", "World")val clazz getClassFromList(list)println(clazz)…...

AQS源码解析

关于 AQS&#xff0c;网上已经有无数的文章阐述 AQS 的使用及其源码&#xff0c;所以多这么一篇文章也没啥所谓&#xff0c;还能总结一下研究过的源码。源码解析和某某的使用&#xff0c;大概是互联网上 Java 文章中写得最多的主题了。 AQS AQS 是 AbstractQueuedSynchronize…...

关于在VS2017中编译Qt项目遇到的问题

关于在VS2017中编译Qt项目遇到的问题 【QT】VS打开QT项目运行不成功 error MSB6006 “cmd.exe”已退出,代码为 2。如何在VS2017里部署的Qt Designer上编辑槽函数 【QT】VS打开QT项目运行不成功 error MSB6006 “cmd.exe”已退出,代码为 2。 链接 如何在VS2017里部署的Qt Design…...

Python web实战 | 使用 Flask 实现 Web Socket 聊天室

概要 今天我们学习如何使用 Python 实现 Web Socket&#xff0c;并实现一个实时聊天室的功能。本文的技术栈包括 Python、Flask、Socket.IO 和 HTML/CSS/JavaScript。 什么是 Web Socket&#xff1f; Web Socket 是一种在单个 TCP 连接上进行全双工通信的协议。它是 HTML5 中的…...

Android10 Recovery系列(一)隐藏recovery菜单项

一 、背景 起因是遇到了一个隐藏删除recovery菜单项的需求。在寻找解决问题的时候,我经历了找到源码位置,调试修改,生效,思考是否可拓展,优化修改,符合要求的整个过程,下面简单分享一下。如果不想立即实现效果或者只想看解决方案,可以直接看总结那一个部分 二 、准备…...

选好NAS网络储存解决方案,是安全储存的关键

随着网络信息的发展&#xff0c;NAS也越来越受到企业的关注&#xff0c;NAS网络存储除了提供简单的存储服务外&#xff0c;还可以提供更好的数据安全性、更方便的文件共享方式。但市面上的产品种类繁多&#xff0c;我们该如何选择合适的产品&#xff0c;通过企业云盘&#xff0…...

AnimateDiff论文解读-基于Stable Diffusion文生图模型生成动画

文章目录 1. 摘要2. 引言3. 算法3.1 Preliminaries3.2. Personalized Animation3.3 Motion Modeling Module 4. 实验5.限制6. 结论 论文&#xff1a; 《AnimateDiff: Animate Your Personalized Text-to-Image Diffusion Models without Specific Tuning》 github: https://g…...

centos7安装tomcat

安装tomcat 必须依赖 JDK 环境&#xff0c;一定要提前装好JDK保证可以使用 一、下载安装包 到官网下载 上传到linux 服务器 二、安装tomcat 创建tomcat 文件夹 mkdir -p /usr/local/tomcat设置文件夹权限 chmod 757 tomcat将安装包上传至 新建文件夹 解压安装包 tar zx…...

React 第五十五节 Router 中 useAsyncError的使用详解

前言 useAsyncError 是 React Router v6.4 引入的一个钩子&#xff0c;用于处理异步操作&#xff08;如数据加载&#xff09;中的错误。下面我将详细解释其用途并提供代码示例。 一、useAsyncError 用途 处理异步错误&#xff1a;捕获在 loader 或 action 中发生的异步错误替…...

React Native 开发环境搭建(全平台详解)

React Native 开发环境搭建&#xff08;全平台详解&#xff09; 在开始使用 React Native 开发移动应用之前&#xff0c;正确设置开发环境是至关重要的一步。本文将为你提供一份全面的指南&#xff0c;涵盖 macOS 和 Windows 平台的配置步骤&#xff0c;如何在 Android 和 iOS…...

数据链路层的主要功能是什么

数据链路层&#xff08;OSI模型第2层&#xff09;的核心功能是在相邻网络节点&#xff08;如交换机、主机&#xff09;间提供可靠的数据帧传输服务&#xff0c;主要职责包括&#xff1a; &#x1f511; 核心功能详解&#xff1a; 帧封装与解封装 封装&#xff1a; 将网络层下发…...

Neo4j 集群管理:原理、技术与最佳实践深度解析

Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...

ServerTrust 并非唯一

NSURLAuthenticationMethodServerTrust 只是 authenticationMethod 的冰山一角 要理解 NSURLAuthenticationMethodServerTrust, 首先要明白它只是 authenticationMethod 的选项之一, 并非唯一 1 先厘清概念 点说明authenticationMethodURLAuthenticationChallenge.protectionS…...

现代密码学 | 椭圆曲线密码学—附py代码

Elliptic Curve Cryptography 椭圆曲线密码学&#xff08;ECC&#xff09;是一种基于有限域上椭圆曲线数学特性的公钥加密技术。其核心原理涉及椭圆曲线的代数性质、离散对数问题以及有限域上的运算。 椭圆曲线密码学是多种数字签名算法的基础&#xff0c;例如椭圆曲线数字签…...

服务器--宝塔命令

一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行&#xff01; sudo su - 1. CentOS 系统&#xff1a; yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...

Linux离线(zip方式)安装docker

目录 基础信息操作系统信息docker信息 安装实例安装步骤示例 遇到的问题问题1&#xff1a;修改默认工作路径启动失败问题2 找不到对应组 基础信息 操作系统信息 OS版本&#xff1a;CentOS 7 64位 内核版本&#xff1a;3.10.0 相关命令&#xff1a; uname -rcat /etc/os-rele…...

C++课设:简易日历程序(支持传统节假日 + 二十四节气 + 个人纪念日管理)

名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 专栏介绍:《编程项目实战》 目录 一、为什么要开发一个日历程序?1. 深入理解时间算法2. 练习面向对象设计3. 学习数据结构应用二、核心算法深度解析…...

探索Selenium:自动化测试的神奇钥匙

目录 一、Selenium 是什么1.1 定义与概念1.2 发展历程1.3 功能概述 二、Selenium 工作原理剖析2.1 架构组成2.2 工作流程2.3 通信机制 三、Selenium 的优势3.1 跨浏览器与平台支持3.2 丰富的语言支持3.3 强大的社区支持 四、Selenium 的应用场景4.1 Web 应用自动化测试4.2 数据…...