c++のurlmon实现下载文件并进度回调
主文件:
#include "stdafx.h"
#include <UrlMon.h>
#pragma comment(lib, "urlmon.lib")
#include <tchar.h>
#include "cbindCallBack.h"
#include "iostream"
#include <CString> int main()
{ //在url后添加随机数,防止从IE缓存中读取。url后加随机数不会影响下载的。
//如果想要从缓存中提取那么就把下面的注释掉 DWORD rand= GetTickCount(); CBindCallback cbc;
HRESULT hr=URLDownloadToFile(NULL, _T("http://dldir1.qq.com/qqfile/qq/QQ8.9/19983/QQ8.9.exe"), _T("e:\\download\\qq8.9.exe"), NULL, &cbc);
if (hr==S_OK)
{
std::cout << "下载完成!" << std::endl;
system("e:\\download\\qq8.9.exe");
}
else if (hr== E_OUTOFMEMORY)
{
std::cout << "缓冲区长度无效,或内存不足,无法完成操作!" << std::endl;
}
else if (hr== E_OUTOFMEMORY)
{
std::cout << "指定的资源或回调接口无效!" << std::endl;
} getchar();
return ;
} cbindCallback.h #pragma once
#include "stdafx.h"
#include <UrlMon.h>
#pragma comment(lib, "urlmon.lib")
#include <tchar.h>
class CBindCallback : public IBindStatusCallback
{
public:
CBindCallback();
virtual ~CBindCallback(); //接受显示进度窗口的句柄
//CUrlDownloadToFileCallbackTestDlg* m_pdlg; //IBindStatusCallback的方法。除了OnProgress 外的其他方法都返回E_NOTIMPL
STDMETHOD(OnStartBinding)
(DWORD dwReserved,
IBinding __RPC_FAR *pib)
{ return E_NOTIMPL; } STDMETHOD(GetPriority)
(LONG __RPC_FAR *pnPriority)
{ return E_NOTIMPL; } STDMETHOD(OnLowResource)
(DWORD reserved)
{ return E_NOTIMPL; } //OnProgress在这里
STDMETHOD(OnProgress)
(ULONG ulProgress,
ULONG ulProgressMax,
ULONG ulStatusCode,
LPCWSTR wszStatusText); STDMETHOD(OnStopBinding)
(HRESULT hresult,
LPCWSTR szError)
{ return E_NOTIMPL; } STDMETHOD(GetBindInfo)
(DWORD __RPC_FAR *grfBINDF,
BINDINFO __RPC_FAR *pbindinfo)
{ return E_NOTIMPL; } STDMETHOD(OnDataAvailable)
(DWORD grfBSCF,
DWORD dwSize,
FORMATETC __RPC_FAR *pformatetc,
STGMEDIUM __RPC_FAR *pstgmed)
{ return E_NOTIMPL; } STDMETHOD(OnObjectAvailable)
(REFIID riid,
IUnknown __RPC_FAR *punk)
{ return E_NOTIMPL; } // IUnknown方法.IE 不会调用这些方法的 STDMETHOD_(ULONG, AddRef)()
{ return ; } STDMETHOD_(ULONG, Release)()
{ return ; } STDMETHOD(QueryInterface)
(REFIID riid,
void __RPC_FAR *__RPC_FAR *ppvObject)
{ return E_NOTIMPL; }
}; cbindCallBack.cpp #include "stdafx.h"
#include "cbindCallBack.h"
#include "iostream"
using namespace std; //只需实现OnProgress方法,类的实现:
CBindCallback::CBindCallback()
{ } CBindCallback::~CBindCallback()
{ }
//////仅实现OnProgress成员即可
LRESULT CBindCallback::OnProgress(ULONG ulProgress,
ULONG ulProgressMax,
ULONG ulSatusCode,
LPCWSTR szStatusText)
{
/*CProgressCtrl* m_prg = (CProgressCtrl*)m_pdlg->GetDlgItem(IDC_PROGRESS);
m_prg->SetRange32(0, ulProgressMax);
m_prg->SetPos(ulProgress); CString szText;
szText.Format("已下载%d%%", (int)(ulProgress * 100.0 / ulProgressMax));
(m_pdlg->GetDlgItem(IDC_STATUS))->SetWindowText(szText);*/
cout << "文件大小为:" << ulProgressMax // << "MB"<<endl;
cout << ulProgress//<<"MB" << endl;
cout << "已下载:" << ulProgressMax*100.0/ulProgressMax << "%" << endl;
return S_OK;
}
注意事项:
1、下载代码最好放到一个线程里,否则URLDownloadToFile下载过程中等待返回时会阻塞,使UI失去响应。
2、OnProgress返回S_OK表示正常,还可以通过返回E_ABORT使下载中断,所以可以设置个超时时间,如果超时的话,就让OnProgress返回E_ABORT。另外下次再开始从同一个url下载同一个文件时会直接由IE缓存中读取已下载的部分,达到“断点续传”的效果。
3、实际测试过程中发现URLDownloadToFile读IE缓存中已经下载的文件会有很大的安全隐患,如果哪次下载的文件发生问题,那么在不清除缓存的情况下,这个函数以后会一直读取损坏的文件而不重新下载。网上搜了一下解决方案,大概有三种:
a.下载前用FindFirstUrlCacheEntry,FindNextUrlCacheEntry,DeleteUrlCacheEntry清除cache,这个代码网上很多。
b.重载IBindStatusCallback的GetBindInfo方法,指定BINDF_GETNEWESTVERSION和BINDF_NOWRITECACHE属性,但是我测试发现即使指定这两个属性UrlDownloadToFile还是会很执着的读缓存,郁闷。
c.还有一种方法比较猥琐,在要下载的文件地址后加一个随机字符串,这样既不会影响正常下载(下载时会被指向正确的地址)而且由于每次传给URLDownloadToFile的url都不同,在cache中没有地址匹配的文件,所以会重新下载。上面的代码就使用了这种方法,个人感觉比较省事而且经测试有效。
4、CBindCallback有个成员变量用来传递进度条所在的窗口句柄m_pdlg,当然这个也可以用其他方式实现。
5、URLDownloadToFile的好处在于它会自动使用IE的设置,完成下载,不用考虑代理情况。
c++のurlmon实现下载文件并进度回调的更多相关文章
- VC下载文件显示进度条
VC下载文件显示进度条 逗比汪星人2009-09-18上传 by Koma http://blog.csd.net/wangningyu http://download.csdn.net/deta ...
- Android OkHttp + Retrofit 下载文件与进度监听
本文链接 下载文件是一个比较常见的需求.给定一个url,我们可以使用URLConnection下载文件. 使用OkHttp也可以通过流来下载文件. 给OkHttp中添加拦截器,即可实现下载进度的监听功 ...
- VC下载文件 + 显示进度条
在codeproject里找了许久,发现这样一个VC下载文件并显示进度条的源码,于是添加了些中文注释: 1.下载线程函数: UINT DownloadFile(LPVOID pParam) { CWn ...
- UrlDownloadFile, 线程下载文件, 带进度条
unit FileDownLoadThread; interface uses Classes, SysUtils, Windows, ActiveX, UrlMon; const S_ABORT = ...
- webclient下载文件 带进度条
private void button1_Click(object sender, EventArgs e) { doDownload(textBox1.Text.Trim()); } private ...
- winform 下载文件显示进度和百分比
/// <summary> /// 下载完成 /// </summary> private void DownloadFileCompleted() { IsComlate = ...
- libcurl开源库在Win32程序中使用下载文件显示进度条实例
一.配置工程引用libcurl库 #define CURL_STATICLIB #include "curl/curl.h" #ifdef _DEBUG #pragma comme ...
- C#使用七牛云存储上传下载文件、自定义回调
项目需要将音视频文件上传服务器,考虑并发要求高,通过七牛来实现. 做了一个简易的压力测试,同时上传多个文件,七牛自己应该有队列处理并发请求,我无论同时提交多少个文件,七牛是批量一个个排队处理了. 一个 ...
- Android 下载文件 显示进度条
加入两个权限 一个是联网,另一个是读写SD卡 <uses-permission android:name="android.permission.INTERNET">& ...
随机推荐
- IdentityServer4 中文文档 -5- (简介)支持和咨询选项
IdentityServer4 中文文档 -5- (简介)支持和咨询选项 原文:http://docs.identityserver.io/en/release/intro/support.html ...
- (转)Visual Studio 2013新功能预览:增代码的透明度和可追溯性
微软打破了Visual Studio两年升级一次的传统,Visual Studio 2012发布还不足一年,微软就计划发布了Visual Studio 2013了.在今天的TechEd大会上,微软宣布 ...
- c# 匿名方法几种表现形式
delegate int del(int a); static void Main(string[] args) { //匿名方法的几种表现形式 del del = delegate (int x) ...
- ECharts图表实战经验1:如何设置图表同序列不同数据点的独立颜色值
最近有不少朋友在追问这样一个问题:我单序列的柱状图,我想让每一个根柱子的颜色都不一样,应该如何做? 针对这个问题,其实我只想说你压根没有认真看完或者查找ECharts官方的示例,官方能够找到的示例有: ...
- [PHP] 算法-有序数组旋转后寻找最小值的PHP实现
把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转. 输入一个非减排序的数组的一个旋转,输出旋转数组的最小元素. 例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组 ...
- 怎样删除C/C++代码中的所有注释?浅谈状态机的编程思想
K&R习题1-23中,要求“编写一个程序,删除C语言程序中所有的注释语句.要正确处理带引号的字符串与字符常量.在C语言中,注释不允许嵌套”. 如果不考虑字符常量和字符串常量,问题确实很简单.只 ...
- Javassm连接数据库报错129 ERROR [com.alibaba.druid.pool.DruidDataSource] - {dataSource-1} init error
Javassm连接数据库报错129 ERROR [com.alibaba.druid.pool.DruidDataSource] - {dataSource-1} init error 发现jdbc这 ...
- Placement of class definition and prototype
When I create a function, I can put the code for it after main if I put the prototype above main. Fo ...
- API接口规范V1.0——制定好规范,才好合作开发
返回码规范: 统一六位 000000 表示成功! 参数相关返回码预留100000-199999:系统相关返回码预留200000-299999:数据中心310000-319999后续项目以此类推,后续根 ...
- react中这些细节你注意过没有?
react中的一些细节知识点: 1.组件中get的使用(作为类的getter) ES6知识:class类也有自己的getter和setter,写法如下: Class Component { const ...