QT 多线程下载和断点续传

学习http,用Qt做了个下载器,支持多线程下载和断点续传。

大致界面如下:

image

主要用到的QT的一下三个关于http的类:
QNetworkAccessManager:http请求的封装函数。主要函数有get,post,put等等用于发送http的请求
QNetworkRequest:设置http请求信息,比如:网址,头部格式等等
QNetworkReply:发送请求后返回的封装类。

官方手册的实例代码很清晰的反映了三者的关系:

QNetworkRequest request;
request.setUrl(QUrl("http://qt-project.org"));
request.setRawHeader("User-Agent", "MyOwnBrowser 1.0");

QNetworkReply *reply = manager->get(request);
connect(reply, &QIODevice::readyRead, this, &MyClass::slotReadyRead);
connect(reply, QOverload<QNetworkReply::NetworkError>::of(&QNetworkReply::error),
      this, &MyClass::slotError);
connect(reply, &QNetworkReply::sslErrors,
      this, &MyClass::slotSslErrors);

三者结合能够很轻松的实现下载功能。在此基础上加上断点续传和多进程下载。

这里所谓的多进程下载,是指将文件分割成多分,分段进行下载,速度经过测试有提升。

需要注意的是两点:

1:不管是多进程下载还是断点续传,都有一个前提条件:请求的服务器支持断点续传功能。不过现如今大多数服务器都支持此功能。
2:"多进程"下载其实在功能实现上来讲就是断点续传

假设一个文件有1024000字节大小;
正常下载文件时可以设置请求头(也可以不设置),比如:

request.setRawHeader(QByteArray("Range"), QString("bytes=0-").toLocal8Bit());

就能下载整个文件。

断点续传:一个文件在下载或上传时,某些人为或非人为情况下中断了,下次可以继续下载或上传。
假如现在下载到156000的时候中断了,想要接着下载就只需要做些许改动便可:

request.setRawHeader(QByteArray("Range"), QString("bytes=156000-").toLocal8Bit());

这时候就开始从156000字节开始下载了。

"多进程"下载:
由断点续传可知,只要将文件大小分成几份分开下载即可。1024000字节大小分成4分,给多个请求即可。
Range:bytes=0-256000
Range:bytes=256000-521000
Range:bytes=521000-768000
Range:bytes=768000-1024000

void DownloadWidget::startDownLoad()
{
    // 分配了另外的m_preReply 获取文件的总大小
    qint64 totalBytes = m_preReply->rawHeader("Content-Length").toLongLong();
    qDebug() << "total size: " << totalBytes;

    // 多线程数量
    int tNum = 5;

    QVector<QVector<qreal>> vecSize(tNum);

    qreal part = totalBytes / tNum;
    for (int i = 0; i < tNum; ++i) {
        QVector<qreal> size;
        size.append(i * part);
        size.append(i * part + part - 1);
        vecSize.append(size);
    }

    //余数部分加入最后一个
    vecSize[tNum -1].at(1) = totalBytes;
    qDebug() << vecSize;

    for (int i = 0; i < tNum; ++i) {
        qint64 bBytes = vecSize.at(i).at(0);
        qint64 eBytes = vecSize.at(i).at(1);

        qDebug() << i << bBytes << eBytes;
        QNetworkRequest request(url);
        request.setRawHeader(QByteArray("Range"), QString("bytes=%1-%2").arg(bBytes).arg(eBytes).toLocal8Bit());

        QNetworkAccessManager *manager = new QNetworkAccessManager(this);
        QNetworkReply *reply = manager->get(request);

        connect(reply, &QNetworkReply::finished, this, &DownloadWidget::httpFinished);  //传输完成发送了QNetworkReply::finished信号
        connect(reply, &QNetworkReply::readyRead, this, &DownloadWidget::httpReadyRead);  //服务器传回一次数据,发送一次QNetworkReply::readyRead
    }
}

多进程的文件下载设置断点续传:
采用xml的形式记录开始,结束,断点三个位置,再次下载时读取数据设置好位置即可。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。