TCP传输文件
传输文件和传输信息的区别:
- 传输信息,只是一条数据,传输文件是多条数据
- 传输信息传输过去一般都会显示,传输文件一般不会显示,一般只是存放在文件中
- 传输文件需要传输,文件大小和文件名称(不然不知道什么时候结束)
在这里使用的是服务器端发送数据,客户端接收数据。
为了在本地测试把服务器端和客户端分项目编写,方便在一台电脑上测试。
总的代码会在文件最后给出
目录显示:
- File_Tcp为服务器端,发送文件
- File_Tcp_Socket 为客户端,接收文件(创建的时候忘记该类名了,这里就使用widget了)
服务器端的编写
发送的规则:
- 先发送,文件的参数,文件大小和文件名
- 然后再发送,文件的内容
创建一个项目;
pro文件中添加:
QT +=network
ui界面添加以下控件:
tcp_server.h文件内容解析
1.添加以下头文件:
#include<QTcpServer>
#include<QMessageBox>
#include<QFile>
#include<QMessageBox>
#include<QTcpSocket>
#include<QHostAddress>
#include<QNetworkInterface>
#include<QFileDialog>
#include<QDataStream>
#include<QFileInfo>
2.添加私有成员:
private:QTcpServer *server;//服务器端QTcpSocket *socket;//套接字QString FilePathName;//文件路径QString FileName;//文件名QFile * LocalFile;//文件指针,打开需要传输的文件qint64 FileSize=0;//文件总大小qint64 TFSize=0;//已传输的大小qint64 ToBeSize=0;//未传输的数据qint64 payloadSize=64*1024;//有效的载荷,64KBQByteArray block;//缓存,用来临时存储需要发送的信息
3.槽函数
private slots:void on_pushButton_3_clicked();void on_pushButton_clicked();void on_pushButton_2_clicked();void on_pushButton_4_clicked();//上面是ui界面转到槽生成的信号void Send_File(qint64 num);//用来传输文件内容
4.事件和其他函数:
protected:void closeEvent(QCloseEvent *event);//重写关闭事件QString getIp();//获取IP
tcp_server.cpp文件内容解析
1.构造函数中的内容
server=new QTcpServer(this);//QTcpServer的初始化QString Ip=getIp();//获取本机的IP地址ui->lineEdit_4->setText(tr("%1").arg(Ip));//再主机的lineEdit中显示IP地址ui->lineEdit_5->setText("6666");//设置端口号socket=nullptr;//初始化QTcpSocketconnect(server,&QTcpServer::newConnection,[=]()//当有新的连接时{QMessageBox::information(this,"提示信息","已有新连接",QMessageBox::Ok);socket=new QTcpSocket(this);//套接字socket=server->nextPendingConnection();//获取套接字});//实现可以多次传输,当文件地址改变时,这些参数都置为0connect(ui->lineEdit,&QLineEdit::textChanged,[=]()//{FileSize=0;//文件总大小TFSize=0;//已传输的大小ToBeSize=0;//未传输的数据ui->lineEdit_2->setText("");ui->lineEdit_3->setText("");});
2.getIP()函数的实现
QString TCP_Server::getIp()//获取IP
{QList<QHostAddress> addss=QNetworkInterface::allAddresses();//获取全部地址foreach(QHostAddress add,addss)//遍历这些地址{if(add.protocol()==QAbstractSocket::IPv4Protocol)//如果地址为IVP4的话{return add.toString();//返回该地址}}return 0;//没有的话返回0
}
3.监听的实现:
void TCP_Server::on_pushButton_3_clicked()//监听
{if(!server->listen(QHostAddress(ui->lineEdit_4->text()),ui->lineEdit_5->text().toInt()));//如果监听失败{return;}
}
4.获取文件名的实现:
void TCP_Server::on_pushButton_clicked()//获取文件名
{FilePathName=QFileDialog::getOpenFileName(this);//获取路径if(!FilePathName.isEmpty())//文件名不为空{QFileInfo fi(FilePathName);//创建一个文件信息对象FileName=fi.fileName();//获取文件名ui->lineEdit->setText(FilePathName);//放置文件名}}
5.取消的实现:
void TCP_Server::on_pushButton_4_clicked()//取消
{if(server->isListening())//如果正在监听{server->close();if(LocalFile->isOpen()){LocalFile->close();}socket->abort();//断开连接}
}
6.重写关闭事件:
void TCP_Server::closeEvent(QCloseEvent *event)//重写关闭事件
{on_pushButton_4_clicked();//取消close();
}
7.发送的实现,(发送文件的一些参数)
void TCP_Server::on_pushButton_2_clicked()//发送文件信息
{LocalFile=new QFile(FilePathName);//打开该文件if(!LocalFile->open(QFile::ReadOnly))//当文件打不开{QMessageBox::information(this,"提示信息","发送失败",QMessageBox::Ok);return;}//当接收完一次数据后,执行传输文件的操作,直到传输完connect(socket,SIGNAL(bytesWritten(qint64)),this,SLOT(Send_File(qint64)));//使用缓存FileSize=LocalFile->size();//获取文件的大小QDataStream Ds(&block,QIODevice::WriteOnly);//创建一个数据流Ds.setVersion(QDataStream::Qt_5_9);//设置版本号;Ds<<qint64(0)<<qint64(0)<<FileName;//将数据写入流中FileSize+=block.size();//获取流中数据的大小Ds.device()->seek(0);//回到开头Ds<<FileSize<<qint64(block.size()-sizeof(qint64)*2);//数据的总长度和文件长度//把文件头传输出去,同时修改待发送的字节数ToBeSize=FileSize-socket->write(block);block.resize(0);//把缓存置0,以便下次使用}
这里主要解释一下
//当接收完一次数据后,执行传输文件的操作,直到传输完connect(socket,SIGNAL(bytesWritten(qint64)),this,SLOT(Send_File(qint64)));
在这里,使用新版的信号于槽,会报错,不知道什么原因。
bytesWritten(qint64):每次将数据有效负载写入设备的当前写入通道时,都会发出此信号。会返回写入的字节数。
当写入通道时,使用Send_File(qint64)来统计字节数
8.Send_File(qint64)的实现
- 首先要统计数据
- 判断数据,如果还有未传输的数据,继续传输
- 在显示总的文件大小和已传输的大小
- 如果已发送的文件大小==总文件大小,关闭文件关闭服务器,提示传输完成
void TCP_Server::Send_File(qint64 num)//传输文件数据
{qApp->processEvents();//防止传输文件时页面冻结TFSize+=(int)num;//更新已传输的数据大小if(ToBeSize>0)//如果未传输的数据大于0{//当剩下的数据小于payloadSize时,优先使用更小的值block=LocalFile->read(qMin(ToBeSize,payloadSize));//读取数据ToBeSize-=(int)socket->write(block,block.size());//发送并更新数据大小block.resize(0);//清空缓存}else{LocalFile->close();}ui->lineEdit_3->setText(tr("%1").arg(FileSize/(1024*1024))+"MB");//总的数据量ui->lineEdit_2->setText(tr("%1").arg(TFSize/(1024*1024))+"MB");//已传输的数据if(FileSize==TFSize){LocalFile->close();//关闭文件server->close();//关闭服务器QMessageBox::information(this,"提示信息","传输完成",QMessageBox::Ok);}}
main函数:
#include "tcp_server.h"
#include <QApplication>int main(int argc, char *argv[])
{QApplication a(argc, argv);TCP_Server w;w.show();return a.exec();
}
服务器端的搭建
接收文件时,需要判断是否接受了文件信息
- 接受了话再接受文件的内容
创建一个项目:
pro文件中添加:
QT +=network
ui界面中添加以下控件:
widget.h文件的内容解析
1.添加以下头文件
#include<QTcpSocket>
#include<QDataStream>
#include<QMessageBox>
#include<QHostAddress>
#include<QFile>
2.添加私有成员
QTcpSocket *socket;//套接字QFile *file;//文件指针qint64 RCDsize;//接收的数据qint64 ToBoSize;//未接收的数据qint64 Filesize;//文件总大小qint64 Allsize;//总大小QString FileName;//文件名称QByteArray inblock;//缓存
3.槽函数
private slots:void on_pushButton_clicked();void on_pushButton_2_clicked();
//以上未ui界面中转到槽实现void readData();//读取数据
4.重写事件
protected:void closeEvent(QCloseEvent *event);//重写关闭事件
widget.cpp文件的内容解析:
1.构造函数
socket=new QTcpSocket(this);ui->lineEdit_2->setText("6666");//设置端口号未6666ToBoSize=0;Filesize=0;RCDsize=0;connect(socket,&QTcpSocket::readyRead,this,&Widget::readData);//读取数据
2.连接的实现
void Widget::on_pushButton_clicked()//连接
{
socket->abort();//断开已有连接//创建连接socket->connectToHost(QHostAddress(ui->lineEdit->text()),ui->lineEdit_2->text().toInt());
}//QHostAddress(ui->lineEdit->text())为地址
//ui->lineEdit_2->text().toInt()为端口号
3.取消的实现
void Widget::on_pushButton_2_clicked()//取消
{socket->abort();//断开连接if(file->isOpen()){file->close();}}
4.重写关闭事件
void Widget::closeEvent(QCloseEvent *event)
{socket->abort();if (file->isOpen())file->close();close();
}
5.读取数据的实现
void Widget::readData()//读取数据
{QDataStream in(socket);//流读取数据in.setVersion(QDataStream::Qt_5_9);if(RCDsize<=sizeof(qint64)*2){if((socket->bytesAvailable()>=sizeof(qint64)*2)&&(Filesize==0)){in>>Allsize>>Filesize;//读取资源的大小,文件的大小RCDsize+=sizeof(qint64)*2;//更新读取的数据}if((socket->bytesAvailable()>=Filesize)&&(Filesize!=0)){in>>FileName;//读取文件名称file=new QFile(FileName);//打开文件ui->lineEdit_3->setText(QString("%1").arg(FileName));RCDsize+=Filesize;//更新读取的数据if(!file->open(QFile::WriteOnly)){QMessageBox::information(this,"提示信息","无法读取文件");return;}}else{return;}}if(RCDsize<Allsize)//如果接受的数据小于总数据,继续接受{RCDsize+=socket->bytesAvailable();inblock=socket->readAll();file->write(inblock);inblock.resize(0);}//更新显示的数据ui->lineEdit_4->setText(QString("%1").arg(RCDsize/(1024*1024))+"MB");ui->lineEdit_5->setText(QString("%1").arg(Allsize/(1024*1024))+"MB");if(RCDsize==Allsize)//如果接受的数据等于总数据,传输完成{file->close();socket->close();QMessageBox::information(this,"提示信息","接收完成",QMessageBox::Ok);//将数据清空以便下次使用file=nullptr;//文件指针RCDsize=0;//接收的数据ToBoSize=0;//未接收的数据Filesize=0;//文件总大小Allsize=0;//总大小FileName="";//文件名称inblock=0;//缓存}
}
运行结果:
- 先运行服务器端,再运行客户端
- 客户端和服务器端都输入IP和端口号
- 服务器端选择文件
- 点击发送
1.输入主机和端口号
2.点击连接,然后点击监听
3.选择文件点击发送
数据太小所以0MB
注意:传输完会断开连接,所以还需要点击连接再点击监听
源码已发出:
QtTcp传输文件(简易)-C++文档类资源-CSDN文库
相关文章:

TCP传输文件
传输文件和传输信息的区别: 传输信息,只是一条数据,传输文件是多条数据传输信息传输过去一般都会显示,传输文件一般不会显示,一般只是存放在文件中传输文件需要传输,文件大小和文件名称(不然不知…...

vue3:加载本地图片等静态资源
背景 在我们用 vue2 webpack 的时候,加载图片资源是这样用的: <img :src"require(/assets/test.png)" />这样打包后就会触发 file-loader 打包图片资源,在 dist 文件夹中就可以看到这个图片(如果图片较小会打包…...
工作记录------数据库group_concat函数长度问题
工作记录------group_concat函数长度问题 背景:页面在数据展示时,报错,错误显示:String index out of range: -1 异常信息 java.lang.StringIndexOutOfBoundsException: String index out of range: -1at java.lang.String.sub…...

Python基础语法
1 编程环境 1.1 编译环境 pycharmpython/anaconda 1.2 环境设置 File -> settings -> Project interpreter -> 1.3 Hello world 2 条件判断 2.1 例题 【题1】输入一个年份,判断是否是闰年 ①能被4整除,但不能被100整除; ②能被400整…...

windows环境下安装Nginx及常用操作命令
windows环境下安装Nginx及常用操作命令nginx基本概述基本用途nginx安装nginx基本概述 Nginx (engine x) 是一个高性能的HTTP和反向代理web服务器。基本用途 nginx是一个轻量级高并发服务器,而tomcat并不是。nginx一般被用来做反向代理,将请求转发到应用…...
python excel数据处理?
前段时间做了个小项目,帮个海洋系的教授做了个数据处理的软件。基本的功能很简单,就是对Excel里面的一些数据进行过滤,统计,对多个表的内容进行合并等。之前没有处理Excel数据的经验,甚至于自己都很少用到Excel。记得《…...

Hudi-集成Flink
文章目录集成Flink环境准备sql-client方式启动sql-client插入数据查询数据更新数据流式插入code 方式环境准备代码类型映射核心参数设置去重参数并发参数压缩参数文件大小Hadoop参数内存优化读取方式流读(Streaming Query)增量读取(Increment…...

重新认识 React Hooks useContext
通常来说,React 数据的传递方式都是一层一层把资料 props 传到子层的 就算第二层(Function Component)、第三层(Button Group Compontn) 根本没有用到这个资料,但是为了传到最底层(button) ,每一层还是必须要传props // App.js const App = () => {const [dark, setDark…...

数据库(2)--加深对统计查询的理解,熟练使用聚合函数
一、内容要求 利用sql建立学生信息数据库,并定义以下基本表: 学生(学号,年龄,性别,系号) 课程(课号,课名,学分,学时) 选课࿰…...

stm32f407探索者开发板(十五)——NVIC中断优先级管理
文章目录零、前言一、NVIC中断优先级分组1.1 中断的管理方法1.2 抢占优先级&相应优先级的区别1.3 举例1.4 特别说明1.5 中断优先级分组函数二、NVIC中断优先级设置2.1 中断设置相关寄存器2.2 中断设置优先级2.2.1 中断优先级控制的寄存器组 IP[240]2.2.2 中断使能寄存器组 …...

【Azure 架构师学习笔记】-Azure Logic Apps(6)- Logic Apps调用ADF
本文属于【Azure 架构师学习笔记】系列。 本文属于【Azure Logic Apps】系列。 接上文【Azure 架构师学习笔记】-Azure Logic Apps(5)- 标准和使用量类型的区别 前言 Logic Apps 和 ADF 的搭配使用是常见的组合,它们可以互相弥补各自的不足和…...
python随机获取列表中某一元素
1、利用Python中的random模块中的choice方法 random.choice()可以从任何序列,比如list列表中,选取一个随机的元素返回,可以用于字符串、列表、元组等。 import random arr[1,2,3,4,5,6] print(random.choice(arr))2、利用Python中的random模…...

Nacos微服务笔记
Nacos安装Nacos 的 Github(Tags alibaba/nacos GitHub)下载我们所需的 Nacos 版本,可以选择 windows 或者 Linux。 进入官网,选择合适版本,tar.gz为linux版本,zip为windows版本。下载并解压 nacos-server…...

MAC文件误删怎么办?mac数据恢复,亲测很好用的方法
电脑文件误删,应该很多人都经历过。之前分享了很多关于Windows电脑文件误删如何恢复的方法,那么MAC电脑文件误删该怎么办?有什么好方法可以使得mac数据恢复回来吗?下面就给大家分享一些亲测好用的方法! 一、MAC电脑的文…...

机械革命z2黑苹果改造计划第二番-MacOS实用软件渗透工具
机械革命z2黑苹果改造计划第二番-实用软件 Mac实用工具 这是旧电脑改造计划的第二篇,就是安装一些常用软件和一些渗透测试工具,武装灵魂成为真正的生产力工具 首先推荐一个网站,www.mactools.app,这个软件里边有大多数常用的软…...

【LeetCode】每日一题(4)
目录 题目:1124. 表现良好的最长时间段 - 力扣(Leetcode) 题目的接口: 解题思路: 代码: 过啦!!! 写在最后: 题目:1124. 表现良好的最长时间…...
Linux内核移植:内核的启动过程分析、启动配置与rootfs必要文件
Linux内核移植:内核的启动过程、启动配置与rootfs必要文件一、启动过程二、启动配置(一)SysV初始化(二)systemd初始化三、rootfs中的启动配置文件1、inittab2、/etc/init.d/rcS 脚本3、fstab4、profile 文件5、其他文件…...
【代码随想录训练营】【Day14】第六章|二叉树|理论基础|递归遍历|迭代遍历|统一迭代
理论基础 二叉树的定义形式有:节点指针和数组 在数组中,父节点的下标为i,那么其左孩子的下标即i*21,右孩子的下标即为i*22 二叉树的常见遍历形式有:前序遍历、后序遍历、中序遍历和层序遍历 前序遍历:二…...

AXI-Stream 学习笔记
参考 https://wuzhikai.blog.csdn.net/article/details/121326701 https://zhuanlan.zhihu.com/p/152283168 AXI4 介绍 AXI4 是ARM公司提出的一种片内总线,描述了主从设备之间的数据传输方式。主要有AXI4_LITE、AXI4_FULL、AXI4_STREAM三种。 AXI4_LITE࿱…...

【Linux】程序进程地址空间
文章目录程序地址空间进程地址空间程序地址空间 在Linux下,这种地址叫做 虚拟地址, 我们在用C/C语言所看到的地址,全部都是虚拟地址!物理地址,用户一概看不到,由OS统一管理 问:C/C程序地址空间是内存吗? -> 根本就不是内存! 是进程虚拟地址空间 堆栈…...
SkyWalking 10.2.0 SWCK 配置过程
SkyWalking 10.2.0 & SWCK 配置过程 skywalking oap-server & ui 使用Docker安装在K8S集群以外,K8S集群中的微服务使用initContainer按命名空间将skywalking-java-agent注入到业务容器中。 SWCK有整套的解决方案,全安装在K8S群集中。 具体可参…...

【人工智能】神经网络的优化器optimizer(二):Adagrad自适应学习率优化器
一.自适应梯度算法Adagrad概述 Adagrad(Adaptive Gradient Algorithm)是一种自适应学习率的优化算法,由Duchi等人在2011年提出。其核心思想是针对不同参数自动调整学习率,适合处理稀疏数据和不同参数梯度差异较大的场景。Adagrad通…...

黑马Mybatis
Mybatis 表现层:页面展示 业务层:逻辑处理 持久层:持久数据化保存 在这里插入图片描述 Mybatis快速入门  二、核心功能实现 1. 医院科室展示 /…...
Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务
通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...

linux arm系统烧录
1、打开瑞芯微程序 2、按住linux arm 的 recover按键 插入电源 3、当瑞芯微检测到有设备 4、松开recover按键 5、选择升级固件 6、点击固件选择本地刷机的linux arm 镜像 7、点击升级 (忘了有没有这步了 估计有) 刷机程序 和 镜像 就不提供了。要刷的时…...
C++ 基础特性深度解析
目录 引言 一、命名空间(namespace) C 中的命名空间 与 C 语言的对比 二、缺省参数 C 中的缺省参数 与 C 语言的对比 三、引用(reference) C 中的引用 与 C 语言的对比 四、inline(内联函数…...
Linux云原生安全:零信任架构与机密计算
Linux云原生安全:零信任架构与机密计算 构建坚不可摧的云原生防御体系 引言:云原生安全的范式革命 随着云原生技术的普及,安全边界正在从传统的网络边界向工作负载内部转移。Gartner预测,到2025年,零信任架构将成为超…...
3403. 从盒子中找出字典序最大的字符串 I
3403. 从盒子中找出字典序最大的字符串 I 题目链接:3403. 从盒子中找出字典序最大的字符串 I 代码如下: class Solution { public:string answerString(string word, int numFriends) {if (numFriends 1) {return word;}string res;for (int i 0;i &…...
使用Matplotlib创建炫酷的3D散点图:数据可视化的新维度
文章目录 基础实现代码代码解析进阶技巧1. 自定义点的大小和颜色2. 添加图例和样式美化3. 真实数据应用示例实用技巧与注意事项完整示例(带样式)应用场景在数据科学和可视化领域,三维图形能为我们提供更丰富的数据洞察。本文将手把手教你如何使用Python的Matplotlib库创建引…...