VC++调用libcurl的VC库使用详解

本文介绍如何使用libcurl库进行HTTP及HTTPS请求处理,并提供了一个详细的示例代码,包括库的安装配置、请求发起、响应处理等步骤。

1、下载官方库。

地址:http://curl.haxx.se/download.html#Win32  下载  Win32 - MSVC,下面有两个版本的库,一个是带ssl的,一个是不带ssl的。

不带ssl的:http://curl.haxx.se/download/libcurl-7.18.0-win32-msvc.zip

带ssl的:http://curl.haxx.se/download/libcurl-7.19.3-win32-ssl-msvc.zip

 

2、在VS2010中VC++目录中加入从zip中解压的include目录,和lib目录

 

3、使用方法为,注意还有debug和Release两种库

#include <curl/curl.h>
//不带SSL
#pragma comment(lib, "libcurl.lib")
//带SSL
#pragma comment(lib, "libcurl_imp.lib")
 

4、 下面是我写的用于http和https的学习记录

// libcurl http和https学习记录 2012-6-30 by Dewei
//

#include "stdafx.h"
#include <string>
#include <iostream>
#include <assert.h> 
#include "curl/curl.h"
#pragma comment(lib, "libcurl_imp.lib")

#define  SKIP_PEER_VERIFICATION 1
//#define  SKIP_HOSTNAME_VERFICATION 1

/*
ptr是指向存储数据的指针,
size是每个块的大小,
nmemb是指块的数目,
stream是用户参数。
所以根据以上这些参数的信息可以知道,ptr中的数据的总长度是size*nmemb
*/
size_t call_wirte_func(const char *ptr, size_t size, size_t nmemb, std::string *stream)
{
    assert(stream != NULL);
    size_t len  = size * nmemb;
    stream->append(ptr, len);
    return len;
}
// 返回http header回调函数  
size_t header_callback(const char  *ptr, size_t size, size_t nmemb, std::string *stream)  
{  
    assert(stream != NULL);
    size_t len  = size * nmemb;
    stream->append(ptr, len);
    return len;
}  

int _tmain(int argc, _TCHAR* argv[])
{


    CURL *curl;
    CURLcode code;
    std::string szbuffer;
    std::string szheader_buffer;
    char errorBuffer[CURL_ERROR_SIZE];
    std::string url = "http://www.douban.com";
    //std::string url = "https://vip.icbc.com.cn/icbc/perbank/index.jsp";
    std::string useragent = "Mozilla/5.0 (Windows NT 6.1; WOW64; rv:13.0) Gecko/20100101 Firefox/13.0.1";
    /*
    CURL_GLOBAL_ALL                //初始化所有的可能的调用。
    CURL_GLOBAL_SSL               //初始化支持 安全套接字层。
    CURL_GLOBAL_WIN32            //初始化win32套接字库。
    CURL_GLOBAL_NOTHING         //没有额外的初始化。
    */
    curl_global_init(CURL_GLOBAL_ALL);
    curl = curl_easy_init();
    if(curl) {
        // 远程URL,支持 http, https, ftp
        curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
        curl_easy_setopt(curl, CURLOPT_USERAGENT, useragent.c_str());
        // 官方下载的DLL并不支持GZIP,Accept-Encoding:deflate, gzip
        curl_easy_setopt(curl, CURLOPT_ENCODING, "gzip, deflate");          
        //curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L);//调试信息打开
        //https 访问专用:start
#ifdef SKIP_PEER_VERIFICATION
        //跳过服务器SSL验证,不使用CA证书
        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
        //如果不跳过SSL验证,则可指定一个CA证书目录
        //curl_easy_setopt(curl, CURLOPT_CAPATH, "this is ca ceat");
#endif

#ifdef SKIP_HOSTNAME_VERFICATION
        //验证服务器端发送的证书,默认是 2(高),1(中),0(禁用)
        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L);
#endif
        //https 访问专用:end


        //发送cookie值给服务器
        //curl_easy_setopt(curl, CURLOPT_COOKIE, "name1=var1; name2=var2;"); 
        /* 与服务器通信交互cookie,默认在内存中,可以是不存在磁盘中的文件或留空 */
        curl_easy_setopt(curl, CURLOPT_COOKIEFILE, "./cookie.txt"); 
        /* 与多个CURL或浏览器交互cookie,会在释放内存后写入磁盘文件 */
        curl_easy_setopt(curl, CURLOPT_COOKIEJAR, "./cookie.txt"); 

        /* POST 数据 */
        // curl_easy_setopt(curl, CURLOPT_POSTFIELDS, "name=daniel&project=curl");
        //设置重定向的最大次数
        curl_easy_setopt(curl, CURLOPT_MAXREDIRS, 5);
        //设置301、302跳转跟随location
        curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1);
        //抓取内容后,回调函数
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, call_wirte_func);
        curl_easy_setopt(curl, CURLOPT_WRITEDATA, &szbuffer);
        //抓取头信息,回调函数
        curl_easy_setopt(curl, CURLOPT_HEADERFUNCTION, header_callback );
        curl_easy_setopt(curl, CURLOPT_HEADERDATA, &szheader_buffer);

        /* 
        CURLE_OK	任务完成一切都好
        CURLE_UNSUPPORTED_PROTOCOL	不支持的协议,由URL的头部指定
        CURLE_COULDNT_CONNECT	不能连接到remote 主机或者代理
        CURLE_REMOTE_ACCESS_DENIED	访问被拒绝
        CURLE_HTTP_RETURNED_ERROR	Http返回错误
        CURLE_READ_ERROR	读本地文件错误 
        CURLE_SSL_CACERT	访问HTTPS时需要CA证书路径
        */
        code = curl_easy_perform(curl);
        if(CURLE_OK == code) {
            double val;

            /* check for bytes downloaded */
            code = curl_easy_getinfo(curl, CURLINFO_SIZE_DOWNLOAD, &val);
            if((CURLE_OK == code) && (val>0))
                printf("Data downloaded: %0.0f bytes.\n", val);

            /* check for total download time */
            code = curl_easy_getinfo(curl, CURLINFO_TOTAL_TIME, &val);
            if((CURLE_OK == code) && (val>0))
                printf("Total download time: %0.3f sec.\n", val);

            /* check for average download speed */
            code = curl_easy_getinfo(curl, CURLINFO_SPEED_DOWNLOAD, &val);
            if((CURLE_OK == code) && (val>0))
                printf("Average download speed: %0.3f kbyte/sec.\n", val / 1024);

            printf("%s\n",szbuffer.c_str()); 
        } 
        else {
            fprintf(stderr, "Failed to get '%s' [%s]\n", url, errorBuffer);
            // exit(EXIT_FAILURE);
        }

        /* 释放内存 */
        curl_easy_cleanup(curl);
    }
    curl_global_cleanup();

    getchar();
    return 0;
}
 

 

提示:

若没有找到zlib1.dll文件。

http://gnuwin32.sourceforge.net/downlinks/zlib-bin-zip.php 下载zib包,将dll文件拷贝至debug目录,程序即可正常执行了。

 

版本使用了7.66 没有用最新的,测试的时候发现7.66以后的版本每次访问都会有1~3条本地(127.0.0.1)的连接和发送数据,不知道是什么问题 支持 HTTP/HTTPS/WebSocket/zlib (WebSocket 模拟出来的) 其他的协议都去掉了,SSL由openssl换成SSPI 文件小的很多 修改内容 增加 curl_slist_add 以替代 curl_slist_append curl_slist_append 接口在添加的时候没有检查重复和覆盖的功能 添加相同的头信息后发送的时候会有多条头信息 curl_slist_add  添加的时候会检查重复并覆盖,删除重复的头 添加 Curl_Perform 以替代 curl_easy_perform 可实现提交取返回一次性完成,不需要自己设置回调,处理数据 增加上传/下载进度的功能,可实现获取实时进度 CURL本来不支持 WebSocket 协议 使用了模拟的方法 实现了 WebSocket 协议 使用 CURLOPT_CONNECT_ONLY 连接服务器成功后 构造请求头发送后获取返回的数据,判断是否是 WebSocket 之后创建线程,循环获取服务器返回的数据, 发送和接收到的数据 使用 WebSocket 协议进行组包和拆包,得到实际的数据 访问的 URL,把 WS/WSS 替换成 HTTP/HTTPS 即可 没有做长连接测试,不知道是否支持 WebSocket 只做了简单是处理,可能有BUF,有能力的可以完善下 支持多线程,多线程循环访问没有出错 libcurl.rar (179 K) 下载次数:136 Vc 源码 curl-7.66.rar (3591 K) 下载次数:67 项目在 projects\Windows 文件夹里 我使用的是 Vc14(VS2015),Vc14-是原始未修改的 Vc10(VS2010,生成的文件更小),也是修改的,可以用,Vc10-是原始未修改的
使用 VC++ 结合 Libcurl 实现 GET 请求,可按以下步骤操作: ### 环境配置 在 VC++ 项目中配置 Libcurl ,需要在项目属性中进行如下设置: 1. **包含目录**:将 Libcurl 的头文件所在目录添加到“属性 -> C/C++ -> 常规 -> 附加包含目录”中。 2. **目录**:将 Libcurl文件所在目录添加到“属性 -> 链接器 -> 常规 -> 附加目录”中。 3. **附加依赖项**:在“属性 -> 链接器 -> 输入 -> 附加依赖项”中添加 `libcurl_a.lib;ws2_32.lib;winmm.lib;wldap32.lib;Crypt32.lib;Normaliz.lib;` [^3]。 ### 代码实现 以下是一个使用 Libcurl 发送 GET 请求的示例代码: ```cpp #include <iostream> #include <curl/curl.h> // 回调函数,用于处理响应数据 size_t WriteCallback(void *contents, size_t size, size_t nmemb, std::string *s) { size_t newLength = size * nmemb; try { s->append((char*)contents, newLength); } catch(std::bad_alloc &e) { return 0; } return newLength; } int main() { CURL *curl; CURLcode res; std::string readBuffer; // 初始化 Libcurl curl = curl_easy_init(); if(curl) { // 设置请求的 URL curl_easy_setopt(curl, CURLOPT_URL, "http://example.com"); // 设置回调函数 curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteCallback); // 设置存储响应数据的字符串 curl_easy_setopt(curl, CURLOPT_WRITEDATA, &readBuffer); // 执行请求 res = curl_easy_perform(curl); // 检查请求是否成功 if(res != CURLE_OK) { std::cerr << "curl_easy_perform() failed: " << curl_easy_strerror(res) << std::endl; } else { // 输出响应数据 std::cout << readBuffer << std::endl; } // 清理 Libcurl curl_easy_cleanup(curl); } return 0; } ``` ### 代码解释 1. **初始化 Libcurl**:使用 `curl_easy_init()` 函数初始化 Libcurl 会话。 2. **设置请求参数**:使用 `curl_easy_setopt()` 函数设置请求的 URL、回调函数和存储响应数据的字符串。 3. **执行请求**:使用 `curl_easy_perform()` 函数执行请求。 4. **检查请求结果**:检查 `curl_easy_perform()` 函数的返回值,判断请求是否成功。 5. **清理 Libcurl**:使用 `curl_easy_cleanup()` 函数清理 Libcurl 会话。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值