学习http,用Qt做了个下载器,支持多线程下载和断点续传。
大致界面如下:
主要用到的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的形式记录开始,结束,断点三个位置,再次下载时读取数据设置好位置即可。