Qt 使用QNetworkReply或者URLDownloadToFile两种不同方式下载http、https链接资源文件,并且获取下载进度。
目录
- 一、 使用 URLDownloadToFile 下载
- 二、 使用 QNetworkReply 下载
- 三、 打包好的可执行程序示例下载
- 四、 会员或订阅专栏下载源码
一、 使用 URLDownloadToFile 下载
使用【 URLDownloadToFile 】下载数据,只需调用函数,如果需要显示进度信息也只需要继承【IBindStatusCallback】类,通过重写OnProgress方法就能获取到及时进度:
头文件调用:
#include <iostream>
#include <Urlmon.h>
// 下面这个也是
#pragma comment(lib, "Urlmon.lib")
// HRESULT URLDownloadToFile(
// LPUNKNOWN pCaller,
// LPCTSTR szURL,
// LPCTSTR szFileName,
//_Reserved_ DWORD dwReserved,
// LPBINDSTATUSCALLBACK lpfnCB
//);
调用:
HRESULT ret =URLDownloadToFile(nullptr,_httpurl,_filepath,0,(LPBINDSTATUSCALLBACK)this);
if (ret != S_OK)
return false; // 下载失败
继承IBindStatusCallback主要需要实现以下方法,其中通过重写OnProgress方法获取到下载进度,通过返回S_OK或E_ABORT判断是否继续下载文件:
class DOWNFILES:public QObject,public IBindStatusCallback
{
HRESULT __stdcall QueryInterface(const IID &, void **) { return E_NOINTERFACE; }
ULONG STDMETHODCALLTYPE AddRef(void) { return 1; }
ULONG STDMETHODCALLTYPE Release(void) { return 1; }
HRESULT STDMETHODCALLTYPE OnStartBinding(DWORD dwReserved, IBinding *pib) { return E_NOTIMPL; }
virtual HRESULT STDMETHODCALLTYPE GetPriority(LONG *pnPriority) { return E_NOTIMPL; }
virtual HRESULT STDMETHODCALLTYPE OnLowResource(DWORD reserved){ return S_OK; }
virtual HRESULT STDMETHODCALLTYPE OnStopBinding(HRESULT hresult, LPCWSTR szError) { return E_NOTIMPL; }
virtual HRESULT STDMETHODCALLTYPE GetBindInfo(DWORD *grfBINDF, BINDINFO *pbindinfo) { return E_NOTIMPL; }
HRESULT STDMETHODCALLTYPE OnDataAvailable(DWORD grfBSCF, DWORD dwSize, FORMATETC *pformatetc, STGMEDIUM *pstgmed){ return E_NOTIMPL; }
virtual HRESULT STDMETHODCALLTYPE OnObjectAvailable(REFIID riid, IUnknown *punk) { return E_NOTIMPL; }
// virtual HRESULT __stdcall OnProgress(ULONG ulProgress, ULONG ulProgressMax, ULONG ulStatusCode, LPCWSTR szStatusText)
// { if (ulProgressMax != 0) { double *percentage = new double(ulProgress * 1.0 / ulProgressMax * 100); //将percentage 发送给显示 delete percentage; } return S_OK;
///进度
HRESULT OnProgress( unsigned long ulProgress, unsigned long ulProgressMax, unsigned long ulStatusCode, LPCWSTR szStatusText) ;
//{if(!THREAD_IS_SHOULD_QUIT){ return S_OK;} else {return E_ABORT;}}
};
注意: 使用URLDownloadToFile 下载数据时会在
【(参考路径)C:\Users\admin\AppData\Local\Microsoft\Windows\INetCache\IE】目录下生成缓存文件,如果是小文件影响不大,如果超过1G,系统盘很容易就没有使用空间。
也可以使用 DeleteUrlCacheEntry(URL链接) 或删除参考路径下的文件,但可能有其他问题
不建议使用这种方式下载。
二、 使用 QNetworkReply 下载
使用【QNetworkReply】下载数据,主要通过readyRead和downloadProgress信号保存数据和获取进度。如果是https链接还需要下载QSslSocket相关库。
pro头添加:
QT += network
URL下载资源数据校验参考:
可通过修改RawHeader的Range参数实现下载的断点继续下载功能。
通过返回的Last-Modified参考是否是最新文件,
通过返回的Content-Range获取文件大小。
添加QNetworkRequest 的重定向防止部分链接是跳转Https资源
#include <QNetworkRequest>
#include <QNetworkAccessManager>
#include <QNetworkReply>
#include <QTimer>
//! url
//! filesize 获取文件大小
//! outime 链接超时时间
bool QTDOWNFILES::urlData_Validation(QString url,quint64& filesize,int outime)
{
bool IsSuccess=false;
qDebug()<<"[url] 数据效验:";
QNetworkRequest request;
request.setAttribute(QNetworkRequest::FollowRedirectsAttribute, true); //启动重定向
qDebug()<<"[urlData_Validation] "<<url;
///获取前500个字节效验文件是否过期
request.setRawHeader("Range","bytes=0-499");
request.setUrl(QUrl(url));
//超时时间
QTimer * out_timer = new QTimer();
QEventLoop eventLoop;
QNetworkAccessManager* manager=new QNetworkAccessManager();
QNetworkReply *reply =manager->get(request);
///线程超时
QObject::connect(out_timer,&QTimer::timeout,reply,&QNetworkReply::abort,Qt::UniqueConnection);
///只需要加载一次
QObject::connect(reply, SIGNAL(readyRead()),&eventLoop, SLOT(quit()));
out_timer->start(outime);
eventLoop.exec(QEventLoop::ExcludeUserInputEvents);
out_timer->stop();
if(reply->error() != QNetworkReply::NoError)
{
IsSuccess=false;
}
if(reply->hasRawHeader("Content-Range"))
{
QVariant contentrange = reply->rawHeader("Content-Range");
qDebug()<<"[Content-Range] "<<contentrange;
QString size=contentrange.toString().mid((0,contentrange.toString().indexOf("/")+1));
qDebug()<<"Size : "<<size;
filesize=size.toLongLong();
IsSuccess=true;
}
// if(reply->hasRawHeader("Last-Modified"))
// {
// QVariant contentrange = reply->rawHeader("Last-Modified");
// QString LastModified=contentrange.toString();
// qDebug()<<"LastModified : "<<LastModified;
// }
manager->clearConnectionCache();
manager->clearAccessCache();
reply->abort();
reply->close();
return IsSuccess;
}
绑定readyRead信号获取数据,和获取进度:
void QTDOWNFILES::WriteFileData()
{
if(reply!=nullptr && file!=nullptr)
{
//file 打开的QFILE类型文件
QByteArray temparray= reply->readAll();
if(temparray.length()>0)
file->write(temparray);
temparray.clear();
}
}
void QTDOWNFILES::HaveFulfilled(qint64 bytesRead,qint64 totalBytes)
{
emit ToFulfilled(bytesRead,totalBytes);
}
1.注意: 使用QNetworkReply 下载时为了防止界面卡顿而使用多线程时,可以将QNetworkReply 的方法实现放入QOBJECT类中,在使用一个线程类来调用QOBJECT类中的方法,这样就避免了写入文件线程不同步的问题。也避免使用写锁。
2.SSL问题: HTTPS资源文件需要使用SSL库。
否则出现异常提示:qt.network.ssl: QSslSocket::connectToHostEncrypted: TLS initialization failed
参考链接:
1.Qt 解决qt.network.ssl: QSslSocket::connectToHostEncrypted: TLS initialization failed问题
2.解决qt.network.ssl: QSslSocket::connectToHostEncrypted: TLS initialization failed
三、 打包好的可执行程序示例下载
详见文章绑定资源;
四、 会员或订阅专栏下载源码
Qt开发项目案例-以及部分示例的源码下载链接