如何在 C/C++ 中使用 WinHTTP 下载文件?

Posted

技术标签:

【中文标题】如何在 C/C++ 中使用 WinHTTP 下载文件?【英文标题】:How to download a file with WinHTTP in C/C++? 【发布时间】:2010-10-23 18:57:29 【问题描述】:

我知道如何下载 html/txt 页面。例如:

//Variables 
DWORD dwSize = 0;
DWORD dwDownloaded = 0;
LPSTR pszOutBuffer;
vector <string>  vFileContent;
BOOL  bResults = FALSE;
HINTERNET  hSession = NULL, 
           hConnect = NULL,
           hRequest = NULL;

// Use WinHttpOpen to obtain a session handle.
hSession = WinHttpOpen( L"WinHTTP Example/1.0",  
                        WINHTTP_ACCESS_TYPE_DEFAULT_PROXY,
                        WINHTTP_NO_PROXY_NAME, 
                        WINHTTP_NO_PROXY_BYPASS, 0);

// Specify an HTTP server.
if (hSession)
    hConnect = WinHttpConnect( hSession, L"nytimes.com",
                               INTERNET_DEFAULT_HTTP_PORT, 0);

// Create an HTTP request handle.
if (hConnect)
    hRequest = WinHttpOpenRequest( hConnect, L"GET", L"/ref/multimedia/podcasts.html",
                                   NULL, WINHTTP_NO_REFERER, 
                                   NULL, 
                                   NULL);

// Send a request.
if (hRequest)
    bResults = WinHttpSendRequest( hRequest,
                                   WINHTTP_NO_ADDITIONAL_HEADERS,
                                   0, WINHTTP_NO_REQUEST_DATA, 0, 
                                   0, 0);


// End the request.
if (bResults)
    bResults = WinHttpReceiveResponse( hRequest, NULL);

// Keep checking for data until there is nothing left.
if (bResults)
    do 
    

        // Check for available data.
        dwSize = 0;
        if (!WinHttpQueryDataAvailable( hRequest, &dwSize))
            printf( "Error %u in WinHttpQueryDataAvailable.\n",
                    GetLastError());

        // Allocate space for the buffer.
        pszOutBuffer = new char[dwSize+1];
        if (!pszOutBuffer)
        
            printf("Out of memory\n");
            dwSize=0;
        
        else
        
            // Read the Data.
            ZeroMemory(pszOutBuffer, dwSize+1);

            if (!WinHttpReadData( hRequest, (LPVOID)pszOutBuffer, 
                                  dwSize, &dwDownloaded))
            
                printf( "Error %u in WinHttpReadData.\n", 
                        GetLastError());
            
            else
            
                        printf("%s", pszOutBuffer);
                            // Data in vFileContent
                vFileContent.push_back(pszOutBuffer);
            

            // Free the memory allocated to the buffer.
            delete [] pszOutBuffer;
        

     while (dwSize>0);


// Report any errors.
if (!bResults)
    printf("Error %d has occurred.\n",GetLastError());

// Close any open handles.
if (hRequest) WinHttpCloseHandle(hRequest);
if (hConnect) WinHttpCloseHandle(hConnect);
if (hSession) WinHttpCloseHandle(hSession);

// Write vFileContent to file
ofstream out("test.txt",ios::binary);
for (int i = 0; i < (int) vFileContent.size();i++)
out << vFileContent[i];
out.close();

当我尝试下载图片时,我只收到文件的第一行,没有错误消息。该问题似乎与 WinHttpOpenRequest 函数中的此参数(ppwszAcceptTypes)有关。

link text

【问题讨论】:

对你的动态内存分配做一点评论——你像pszOutBuffer = new char[dwSize+1];一样分配它,然后检查指针。您不应该这样做,因为 new 运算符默认情况下会在内存不足的情况下抛出异常。为防止出现这种情况,您应该像 pszOutBuffer = new(std::nothrow) char[dwSize+1]; 那样将 std::nothrow 提供给 new 运算符 - 在这种情况下,如果内存不足,指针将为空。 导入 msxml6 库。它附带(可能是vista?肯定是win7,并且可以安装在XP上。启动 COM 并创建 IXMLHTTPRequest 对象并发送请求。比 WinHttp 库和杂耍证书容易得多。 msdn.microsoft.com/en-us/library/ms759148(v=vs.85).aspx 【参考方案1】:

看起来MSDN上的这个帖子是一样的,有解决办法

http://social.msdn.microsoft.com/forums/en-US/vclanguage/thread/45ccd91c-6794-4f9b-8f4f-865c76cc146d

【讨论】:

我以前读过这个帖子,但解决方案(以二进制模式打开文件)对我不起作用。但是我刚刚重试了它,这次使用的是 C,而不是 C++,它可以工作。据我所知,ofstream 不能与 WinHttp 一起使用。【参考方案2】:

解决方案:

FILE * pFile; // NEW
pFile = fopen("file.bin", "w+b"); // NEW

if (bResults)
    do 
    
        // Check for available data.
        dwSize = 0;
        if (!WinHttpQueryDataAvailable( hRequest, &dwSize))
            printf( "Error %u in WinHttpQueryDataAvailable.\n",
                    GetLastError());

        // Allocate space for the buffer.
        pszOutBuffer = new char[dwSize+1];



        if (!pszOutBuffer)
        
            printf("Out of memory\n");
            dwSize=0;
        
        else
        
            // Read the Data.
            ZeroMemory(pszOutBuffer, dwSize+1);

            if (!WinHttpReadData( hRequest, (LPVOID)pszOutBuffer, 
                                  dwSize, &dwDownloaded))
            
                printf( "Error %u in WinHttpReadData.\n", 
                        GetLastError());
            
            else
            
                            printf("%s", pszOutBuffer);
                fwrite(pszOutBuffer, (size_t)dwDownloaded, (size_t)1, pFile); // NEW

            

            // Free the memory allocated to the buffer.
            delete [] pszOutBuffer;
        

     while (dwSize>0);

fclose (pFile); // NEW

【讨论】:

【参考方案3】:

仅仅以二进制模式打开 ofstream 不会改变

【讨论】:

以上是关于如何在 C/C++ 中使用 WinHTTP 下载文件?的主要内容,如果未能解决你的问题,请参考以下文章

C++ winhttp 实现文件下载器

如何从 WinHttp 请求中获取 HTTP 状态码?

有没有在 Windows 上用 C 语言使用 winhttp 的完整例子?

VBA WinHTTP 从受密码保护的 https 网站下载文件

WinHttp 不会从 WinXP 上的 Amazon S3 下载

如何异步使用“WinHttp.WinHttpRequest.5.1”?