QT实现文件传输功能

 更新时间:2022年08月18日 16:30:31   作者:我是黏黏虫  
这篇文章主要为大家详细介绍了QT实现文件传输功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了QT实现文件传输功能的具体代码,供大家参考,具体内容如下

过程如下:

1、服务器端设置监听套接字,开始监听;

2、客户端在连接成功时开始传送文件,有connected()信号连接send()槽,send()发送文件头信息,包括文件名、文件总大小和文件名大小等;

3、传送完文件头信息时开始传送文件内容,有bytesWritten(qint64)信号连接到goOnSend(qint64)槽,前者是当想套接字写入数据时会出发的信号,即当已经想套接字写入数据,就继续传送数据,有send()传送文件头信息开始触发,直到文件传完为止。

4、在服务器端,首先接收文件头信息,然后读取文件名来用新建文件的方式打开一个文件,然后读取文件名即文件等大小信息,用来更新进度条和继续接收数据;

5、实现循环传输,在客户端,因为第一次send()是由connected()信号触发的,之后的每次传送应该手动调用send();在服务器端,在有新数据到达时,会判断是否为头文件,因此在每次文件传完的时候将byteReceived重置为0,即下一次再接收到数据的时候依据byteReceived判断的结果就是一个新文件了。

客户端代码:

#ifndef WIDGET_H  
#define WIDGET_H  
  
#include <QWidget>  
#include <QTcpSocket>  
#include <QFile>  
#include <string>  
  
namespace Ui {  
class Widget;  
}  
  
class Widget : public QWidget  
{  
    Q_OBJECT  
  
public:  
    explicit Widget(QWidget *parent = 0);  
    ~Widget();  
      
private:  
    Ui::Widget *ui;  
  
    QTcpSocket *tcpClient;  
    QFile *localFile;  
    QString fileName;  //文件名  
  
    QByteArray outBlock;  //分次传  
    qint64 loadSize;  //每次发送数据的大小  
    qint64 byteToWrite;  //剩余数据大小  
    qint64 totalSize;  //文件总大小  
  
    int sendTimes;  //用来标记是否为第一次发送,第一次以后连接信号触发,后面的则手动调用  
  
private slots:  
    void send();  //传送文件头信息  
    void goOnSend(qint64);  //传送文件内容  
    void on_openPushButton_clicked();  
    void on_sendPushButton_clicked();  
};  
  
#endif // WIDGET_H  

widget.cpp

#include "widget.h"  
#include "ui_widget.h"  
#include <QHostAddress>  
#include <QTextCodec>  
#include <QFileDialog>  
  
Widget::Widget(QWidget *parent) :  
    QWidget(parent),  
    ui(new Ui::Widget)  
{  
    ui->setupUi(this);  
  
    ui->progressLabel->hide();  
    QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF8"));
  
    tcpClient = new QTcpSocket(this);  
    sendTimes = 0;  
  
    connect(tcpClient, SIGNAL(connected()), this, SLOT(send()));  //当连接成功时,就开始传送文件  
    connect(tcpClient, SIGNAL(bytesWritten(qint64)), this, SLOT(goOnSend(qint64)));   
}  
  
void Widget::send()  //发送文件头信息  
{  
    byteToWrite = localFile->size();  //剩余数据的大小  
    totalSize = localFile->size();  
  
    loadSize = 4*1024;  //每次发送数据的大小  
  
    QDataStream out(&outBlock, QIODevice::WriteOnly);  
    QString currentFileName = fileName.right(fileName.size() - fileName.lastIndexOf('/')-1);  
  
    out<<qint64(0)<<qint64(0)<<currentFileName;  
  
    totalSize += outBlock.size();  //总大小为文件大小加上文件名等信息大小  
    byteToWrite += outBlock.size();  
  
    out.device()->seek(0);  //回到字节流起点来写好前面连个qint64,分别为总大小和文件名等信息大小  
    out<<totalSize<<qint64(outBlock.size());  
  
    tcpClient->write(outBlock);  //将读到的文件发送到套接字  
  
    ui->progressLabel->show();  
    ui->sendProgressBar->setMaximum(totalSize);  
    ui->sendProgressBar->setValue(totalSize - byteToWrite);  
}  
  
void Widget::goOnSend(qint64 numBytes)  //开始发送文件内容  
{  
    byteToWrite -= numBytes;  //剩余数据大小  
    outBlock = localFile->read(qMin(byteToWrite, loadSize));  
    tcpClient->write(outBlock);  
  
    ui->sendProgressBar->setMaximum(totalSize);  
    ui->sendProgressBar->setValue(totalSize - byteToWrite);  
  
    if(byteToWrite == 0)  //发送完毕  
        ui->sendStatusLabel->setText(tr("文件发送完毕!"));  
}  
  
Widget::~Widget()  
{  
    delete ui;  
}  
  
void Widget::on_openPushButton_clicked()  //打开文件并获取文件名(包括路径)  
{  
    ui->sendStatusLabel->setText(tr("正在打开文件..."));  
    ui->sendProgressBar->setValue(0);  //非第一次发送  
  
    loadSize = 0;  
    byteToWrite = 0;  
    totalSize = 0;  
    outBlock.clear();  
  
    fileName = QFileDialog::getOpenFileName(this);  
    localFile = new QFile(fileName);  
    localFile->open(QFile::ReadOnly);  
  
    ui->sendStatusLabel->setText(tr("已打开文件 %1").arg(fileName));  
}  
  
void Widget::on_sendPushButton_clicked()  
{  
    if(sendTimes == 0)  //只有第一次发送的时候,是发生在连接产生信号connect时  
    {  
        tcpClient->connectToHost(QHostAddress("192.168.1.137"), 1000);  
        sendTimes = 1;    
    }  
    else  
        send();  //第一次发送的时候是由connectToHost出发connect信号才能调用send,第二次之后就需要调用send了  
  
    ui->sendStatusLabel->setText(tr("正在发送文件 %1").arg(fileName));  
}  

服务端代码:

widget.h

#ifndef WIDGET_H  
#define WIDGET_H  
  
#include <QWidget>  
#include <QtNetwork/QTcpServer>  
#include <QtNetwork/QTcpSocket>  
#include <QFile>  
#include <QString>  
  
namespace Ui {  
class Widget;  
}  
  
class Widget : public QWidget  
{  
    Q_OBJECT  
      
public:  
    explicit Widget(QWidget *parent = 0);  
    ~Widget();  
      
private:  
    Ui::Widget *ui;  
  
    QTcpServer *server;  
    QTcpSocket *receivedSocket;  
    QFile *newFile;  
  
    QByteArray inBlock;  
    QString fileName;  
    qint64 totalSize;  //总共需要发送的文件大小(文件内容&文件名信息)  
    qint64 byteReceived;  //已经发送的大小  
  
private slots:  
    void acceptConnection();  
    void readClient();  
    void on_pushButton_clicked();  
};  
  
#endif // WIDGET_H  

widget.cpp

#include "widget.h"  
#include "ui_widget.h"  
#include <QTextCodec>  
  
Widget::Widget(QWidget *parent) :  
    QWidget(parent),  
    ui(new Ui::Widget)  
{  
    ui->setupUi(this);  
    ui->progressLabel->hide();   
    QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF8"));
}  
  
void Widget::acceptConnection()  
{  
    ui->receivedStatusLabel->setText(tr("已连接,正在准备接收文件!"));  
  
    receivedSocket = server->nextPendingConnection();  
    connect(receivedSocket, SIGNAL(readyRead()), this, SLOT(readClient()));  
}  
  
void Widget::readClient()  
{  
    ui->receivedStatusLabel->setText(tr("正在接收文件..."));  
  
    if(byteReceived == 0)  //才刚开始接收数据,此数据为文件信息  
    {  
        ui->receivedProgressBar->setValue(0);  
  
        QDataStream in(receivedSocket);  
        in>>totalSize>>byteReceived>>fileName;    
        newFile = new QFile(fileName);  
        newFile->open(QFile::WriteOnly);  
    }  
    else  //正式读取文件内容  
    {  
        inBlock = receivedSocket->readAll();  
  
        byteReceived += inBlock.size();  
        newFile->write(inBlock);  
        newFile->flush();  
    }   
    ui->progressLabel->show();  
    ui->receivedProgressBar->setMaximum(totalSize);  
    ui->receivedProgressBar->setValue(byteReceived);  
  
    if(byteReceived == totalSize)  
    {  
        ui->receivedStatusLabel->setText(tr("%1接收完成").arg(fileName));  
  
        inBlock.clear();  
        byteReceived = 0;  
        totalSize = 0;  
    }  
}  
  
Widget::~Widget()  
{  
    delete ui;  
}   
void Widget::on_pushButton_clicked()  
{  
    totalSize = 0;  
    byteReceived = 0;  
  
    server = new QTcpServer(this);  
    server->listen(QHostAddress("192.168.1.137"), 1000);  
  
    connect(server, SIGNAL(newConnection()), this, SLOT(acceptConnection()));  
  
    ui->receivedProgressBar->setValue(0);  
    ui->receivedStatusLabel->setText(tr("开始监听..."));  
} 

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C++数组模拟之单链表与双链表和栈和队列的实现过程

    C++数组模拟之单链表与双链表和栈和队列的实现过程

    这篇文章主要介绍了C++数组模拟之单链表与双链表和栈和队列的实现过程,了解内部原理是为了帮助我们做扩展,同时也是验证了一个人的学习能力,如果你想让自己的职业道路更上一层楼,这些底层的东西你是必须要会的,跟随下文来具体了解吧
    2023-02-02
  • C++JSON库CJsonObject详解(轻量简单好用)

    C++JSON库CJsonObject详解(轻量简单好用)

    CJsonObject是基于cJSON全新开发一个C++版的JSON库,CJsonObject的最大优势是轻量简单好用,开发效率极高,对多层嵌套json的读取和生成使用非常简单,喜欢的朋友一起看看吧
    2021-04-04
  • Qt中QGroupBox控件的实现

    Qt中QGroupBox控件的实现

    QGroupBox 是 Qt 框架中一个非常有用的控件,它主要用于组织和管理一组相关的控件,本文主要介绍了Qt中QGroupBox控件的实现,具有一定的参考价值,感兴趣的可以了解一下
    2025-04-04
  • C语言各种符号的使用介绍下篇

    C语言各种符号的使用介绍下篇

    C 语言的基本符号就有 20 多个,每个符号可能同时具有多重含义,而且这些符号之间相互组合又使得 C 语言中的符号变得更加复杂起来
    2022-08-08
  • 举例说明自定义C++异常处理的实例

    举例说明自定义C++异常处理的实例

    这篇文章主要介绍了举例说明自定义C++异常处理的实例的相关资料,这里举例说明该如何使用C++ 的异常,需要的朋友可以参考下
    2017-10-10
  • C++lambda表达式使用介绍

    C++lambda表达式使用介绍

    Lambda 表达式(lambda expression)是一个匿名函数,Lambda表达式基于数学中的λ演算得名。本文就来为大家详细讲讲C++中Lambda表达式的使用,需要的可以参考一下
    2022-08-08
  • C++小知识:C/C++中不要按值传递数组

    C++小知识:C/C++中不要按值传递数组

    今天小编就为大家分享一篇关于C++小知识:C/C++中不要按值传递数组,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01
  • C++中单链表操作的示例代码

    C++中单链表操作的示例代码

    这篇文章主要为大家详细介绍了C++中单链表操作的示例代码,主要包括单链表的创建、元素的插入和删除等。文中的代码简洁易懂,需要的可以参考一下
    2022-10-10
  • C语言操作符进阶教程(表达式求值隐式类型转换方法)

    C语言操作符进阶教程(表达式求值隐式类型转换方法)

    这篇文章主要为大家介绍了C语言操作符进阶教程(表达式求值隐式类型转换方法)
    2022-02-02
  • 用typedef定义类型详细总结

    用typedef定义类型详细总结

    用typedef可以声明各种类型名,但不能用来定义变量,用typedef可以声明数组类型、字符串类型、使用比较方便
    2013-10-10

最新评论