当使用C++做HTTP客户端时,目前通用的做法就是使用libcurl。其官方网站的地址是http://curl.haxx.se/,该网站主要提供了Curl和libcurl。Curl是命令行工具,用于完成FTP, FTPS, HTTP, HTTPS, GOPHER, TELNET, DICT, FILE 以及 LDAP的命令的请求及接收回馈。libcurl提供给开发者,用于使用C++跨平台的开发各种网络协议的请求及响应。里面的文档非常齐全,不过都是英文的。

本文提供最简单的demo使用libcurl开发HttpClient。主要包括同步的HTTP GET、HTTP POST、HTTPS GET、HTTPS POST。

下载libcurl包,如果使用Linux平台,建议下载源文件编译;如果使用Windows平台,建议下载Win32 - MSVC,下载地址是:http://curl.haxx.se/download.html

  1. #ifndef __HTTP_CURL_H__
  2. #define __HTTP_CURL_H__
  3. #include <string>
  4. class CHttpClient
  5. {
  6. public:
  7. CHttpClient(void);
  8. ~CHttpClient(void);
  9. public:
  10. /**
  11. * @brief HTTP POST请求
  12. * @param strUrl 输入参数,请求的Url地址,如:http://www.baidu.com
  13. * @param strPost 输入参数,使用如下格式para1=val1¶2=val2&…
  14. * @param strResponse 输出参数,返回的内容
  15. * @return 返回是否Post成功
  16. */
  17. int Post(const std::string & strUrl, const std::string & strPost, std::string & strResponse);
  18. /**
  19. * @brief HTTP GET请求
  20. * @param strUrl 输入参数,请求的Url地址,如:http://www.baidu.com
  21. * @param strResponse 输出参数,返回的内容
  22. * @return 返回是否Post成功
  23. */
  24. int Get(const std::string & strUrl, std::string & strResponse);
  25. /**
  26. * @brief HTTPS POST请求,无证书版本
  27. * @param strUrl 输入参数,请求的Url地址,如:https://www.alipay.com
  28. * @param strPost 输入参数,使用如下格式para1=val1¶2=val2&…
  29. * @param strResponse 输出参数,返回的内容
  30. * @param pCaPath 输入参数,为CA证书的路径.如果输入为NULL,则不验证服务器端证书的有效性.
  31. * @return 返回是否Post成功
  32. */
  33. int Posts(const std::string & strUrl, const std::string & strPost, std::string & strResponse, const char * pCaPath = NULL);
  34. /**
  35. * @brief HTTPS GET请求,无证书版本
  36. * @param strUrl 输入参数,请求的Url地址,如:https://www.alipay.com
  37. * @param strResponse 输出参数,返回的内容
  38. * @param pCaPath 输入参数,为CA证书的路径.如果输入为NULL,则不验证服务器端证书的有效性.
  39. * @return 返回是否Post成功
  40. */
  41. int Gets(const std::string & strUrl, std::string & strResponse, const char * pCaPath = NULL);
  42. public:
  43. void SetDebug(bool bDebug);
  44. private:
  45. bool m_bDebug;
  46. };
  47. #endif
  1. #include "httpclient.h"
  2. #include "curl/curl.h"
  3. #include <string>
  4. CHttpClient::CHttpClient(void) :
  5. m_bDebug(false)
  6. {
  7. }
  8. CHttpClient::~CHttpClient(void)
  9. {
  10. }
  11. static int OnDebug(CURL *, curl_infotype itype, char * pData, size_t size, void *)
  12. {
  13. if(itype == CURLINFO_TEXT)
  14. {
  15. //printf("[TEXT]%s\n", pData);
  16. }
  17. else if(itype == CURLINFO_HEADER_IN)
  18. {
  19. printf("[HEADER_IN]%s\n", pData);
  20. }
  21. else if(itype == CURLINFO_HEADER_OUT)
  22. {
  23. printf("[HEADER_OUT]%s\n", pData);
  24. }
  25. else if(itype == CURLINFO_DATA_IN)
  26. {
  27. printf("[DATA_IN]%s\n", pData);
  28. }
  29. else if(itype == CURLINFO_DATA_OUT)
  30. {
  31. printf("[DATA_OUT]%s\n", pData);
  32. }
  33. return 0;
  34. }
  35. static size_t OnWriteData(void* buffer, size_t size, size_t nmemb, void* lpVoid)
  36. {
  37. std::string* str = dynamic_cast<std::string*>((std::string *)lpVoid);
  38. if( NULL == str || NULL == buffer )
  39. {
  40. return -1;
  41. }
  42. char* pData = (char*)buffer;
  43. str->append(pData, size * nmemb);
  44. return nmemb;
  45. }
  46. int CHttpClient::Post(const std::string & strUrl, const std::string & strPost, std::string & strResponse)
  47. {
  48. CURLcode res;
  49. CURL* curl = curl_easy_init();
  50. if(NULL == curl)
  51. {
  52. return CURLE_FAILED_INIT;
  53. }
  54. if(m_bDebug)
  55. {
  56. curl_easy_setopt(curl, CURLOPT_VERBOSE, 1);
  57. curl_easy_setopt(curl, CURLOPT_DEBUGFUNCTION, OnDebug);
  58. }
  59. curl_easy_setopt(curl, CURLOPT_URL, strUrl.c_str());
  60. curl_easy_setopt(curl, CURLOPT_POST, 1);
  61. curl_easy_setopt(curl, CURLOPT_POSTFIELDS, strPost.c_str());
  62. curl_easy_setopt(curl, CURLOPT_READFUNCTION, NULL);
  63. curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, OnWriteData);
  64. curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&strResponse);
  65. curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1);
  66. curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, 3);
  67. curl_easy_setopt(curl, CURLOPT_TIMEOUT, 3);
  68. res = curl_easy_perform(curl);
  69. curl_easy_cleanup(curl);
  70. return res;
  71. }
  72. int CHttpClient::Get(const std::string & strUrl, std::string & strResponse)
  73. {
  74. CURLcode res;
  75. CURL* curl = curl_easy_init();
  76. if(NULL == curl)
  77. {
  78. return CURLE_FAILED_INIT;
  79. }
  80. if(m_bDebug)
  81. {
  82. curl_easy_setopt(curl, CURLOPT_VERBOSE, 1);
  83. curl_easy_setopt(curl, CURLOPT_DEBUGFUNCTION, OnDebug);
  84. }
  85. <pre name="code" class="cpp">   curl_easy_setopt(curl, CURLOPT_URL, strUrl.c_str());
  86. curl_easy_setopt(curl, CURLOPT_READFUNCTION, NULL);
  87. curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, OnWriteData);
  88. curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&strResponse);
  89. /**
  90. * 当多个线程都使用超时处理的时候,同时主线程中有sleep或是wait等操作。
  91. * 如果不设置这个选项,libcurl将会发信号打断这个wait从而导致程序退出。
  92. */
  93. curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1);
  94. curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, 3);
  95. curl_easy_setopt(curl, CURLOPT_TIMEOUT, 3);
  96. res = curl_easy_perform(curl);
  97. curl_easy_cleanup(curl);
  98. return res;
  99. }
  100. int CHttpClient::Posts(const std::string & strUrl, const std::string & strPost, std::string & strResponse, const char * pCaPath)
  101. {
  102. CURLcode res;
  103. CURL* curl = curl_easy_init();
  104. if(NULL == curl)
  105. {
  106. return CURLE_FAILED_INIT;
  107. }
  108. if(m_bDebug)
  109. {
  110. curl_easy_setopt(curl, CURLOPT_VERBOSE, 1);
  111. curl_easy_setopt(curl, CURLOPT_DEBUGFUNCTION, OnDebug);
  112. }
  113. curl_easy_setopt(curl, CURLOPT_URL, strUrl.c_str());
  114. curl_easy_setopt(curl, CURLOPT_POST, 1);
  115. curl_easy_setopt(curl, CURLOPT_POSTFIELDS, strPost.c_str());
  116. curl_easy_setopt(curl, CURLOPT_READFUNCTION, NULL);
  117. curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, OnWriteData);
  118. curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&strResponse);
  119. curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1);
  120. if(NULL == pCaPath)
  121. {
  122. curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, false);
  123. curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, false);
  124. }
  125. else
  126. {
  127. //缺省情况就是PEM,所以无需设置,另外支持DER
  128. //curl_easy_setopt(curl,CURLOPT_SSLCERTTYPE,"PEM");
  129. curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, true);
  130. curl_easy_setopt(curl, CURLOPT_CAINFO, pCaPath);
  131. }
  132. curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, 3);
  133. curl_easy_setopt(curl, CURLOPT_TIMEOUT, 3);
  134. res = curl_easy_perform(curl);
  135. curl_easy_cleanup(curl);
  136. return res;
  137. }
  138. int CHttpClient::Gets(const std::string & strUrl, std::string & strResponse, const char * pCaPath)
  139. {
  140. CURLcode res;
  141. CURL* curl = curl_easy_init();
  142. if(NULL == curl)
  143. {
  144. return CURLE_FAILED_INIT;
  145. }
  146. if(m_bDebug)
  147. {
  148. curl_easy_setopt(curl, CURLOPT_VERBOSE, 1);
  149. curl_easy_setopt(curl, CURLOPT_DEBUGFUNCTION, OnDebug);
  150. }
  151. curl_easy_setopt(curl, CURLOPT_URL, strUrl.c_str());
  152. curl_easy_setopt(curl, CURLOPT_READFUNCTION, NULL);
  153. curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, OnWriteData);
  154. curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&strResponse);
  155. curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1);
  156. if(NULL == pCaPath)
  157. {
  158. curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, false);
  159. curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, false);
  160. }
  161. else
  162. {
  163. curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, true);
  164. curl_easy_setopt(curl, CURLOPT_CAINFO, pCaPath);
  165. }
  166. curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, 3);
  167. curl_easy_setopt(curl, CURLOPT_TIMEOUT, 3);
  168. res = curl_easy_perform(curl);
  169. curl_easy_cleanup(curl);
  170. return res;
  171. }
  172. ///////////////////////////////////////////////////////////////////////////////////////////////
  173. void CHttpClient::SetDebug(bool bDebug)
  174. {
  175. m_bDebug = bDebug;
  176. }
http://blog.csdn.net/huyiyang2010/article/details/7664201
 
当POST的内容为para=val,其中val包含'+'的时候会把'+'替换为' ',例如表单中有一项隐藏验证内容

  1. <input type="hidden" name="__VIEWSTATE" id="__VIEWSTATE" value="/wEPDwUJNTc4MTM4MzMzZGQF+FxsPex77Sht1ZxPy5wLSXk1lpdzRqG9qS69Ybnkmg==" />

服务器收到的__VIEWSTATE值为

  1. <br>/wEPDwUJNTc4MTM4MzMzZGQF FxsPex77Sht1ZxPy5wLSXk1lpdzRqG9qS69Ybnkmg==<br>

不知道是不是本来就有这个BUG

Re: xiejienet 2013-12-23 01:03发表 [回复]
回复xingtianduiyue:post和get的数据,最好url编码

C++使用libcurl做HttpClient(业务观摩,用C++封装过程式代码,post和get的数据,最好url编码,否则+会变成空格)good的更多相关文章

  1. C++使用libcurl做HttpClient 和 curl_easy_setopt

    curl_easy_setopt 参数设置 https://curl.haxx.se/libcurl/c/curl_easy_setopt.html  使用libcurl做HttpClient #if ...

  2. vue各生命周期适合做的业务逻辑

    一.实际项目中使用最多的Vue生命周期大概是  created  mounted  updated 二.各自适合做的业务逻辑 1. created   相当于是页面刚开始加载的状态,此时不能操作实例的 ...

  3. 一步步教你为网站开发Android客户端---HttpWatch抓包,HttpClient模拟POST请求,Jsoup解析HTML代码,动态更新ListView

    本文面向Android初级开发者,有一定的Java和Android知识即可. 文章覆盖知识点:HttpWatch抓包,HttpClient模拟POST请求,Jsoup解析HTML代码,动态更新List ...

  4. ASP.NET MVC+EF框架+EasyUI实现权限管理系列(4)-业务逻辑层的封装

    原文:ASP.NET MVC+EF框架+EasyUI实现权限管理系列(4)-业务逻辑层的封装 ASP.NET MVC+EF框架+EasyUI实现权限管系列 (开篇)   (1):框架搭建    (2) ...

  5. Java HttpClient伪造请求之简易封装满足HTTP以及HTTPS请求

    HttpClient简介 HTTP 协议可能是现在 Internet 上使用得最多.最重要的协议了,越来越多的 Java 应用程序需要直接通过 HTTP 协议来访问网络资源.虽然在 JDK 的 jav ...

  6. windows下openresty中使用lua做接口转发、二次封装等

    需求:根据客户需求,可以在ngx下 通过lua做接口二次封装再次转发给用户或第三方 场景:对返回值有要求的.接口屏蔽字段.或做一些业务上的验证等 1.windows直接下载openresty 解压即可 ...

  7. .NET应用架构设计—工作单元模式(摆脱过程式代码的重要思想,代替DDD实现轻量级业务)

    阅读目录: 1.背景介绍 2.过程式代码的真正困境 3.工作单元模式的简单示例 4.总结 1.背景介绍 一直都在谈论面向对象开发,但是开发企业应用系统时,使用面向对象开发最大的问题就是在于,多个对象之 ...

  8. httpclient模拟post请求json封装表单数据

    好长时间不更博了,主要肚子里没什么好墨水,哈哈.废话不说上代码. public static String httpPostWithJSON(String url) throws Exception ...

  9. 4.3.6 对象的界定通过编写接口来访问带这类命名结构的表会出问题。如前所述,SQL Server的灵活性不应用作编写错误代码或创建问题对象的借口。 注意在使用Management Studio的脚本工具时,SQL Server会界定所有的对象。这不是因为这么做是必须的,也不是编写代码的最佳方式,而是因为在界定符中封装所有的对象,比编写脚本引擎来查找需要界定的对象更容易。

    如前所述,在创建对象时,最好避免使用内嵌的空格或保留字作为对象名,但设计人员可能并没有遵守这个最佳实践原则.例如,我当前使用的数据库中有一个审核表名为Transaction,但是Transaction ...

随机推荐

  1. [Compose] 11. Use Task for Asynchronous Actions

    We refactor a standard node callback style workflow into a composed task-based workflow. For example ...

  2. 复制相关参数学习笔记--master上的参数

    特别声明: 所有的过滤规则不建议在主库上设置.     server_id 是一个整数,范围:1 至 power(2,32)-1 之间. 推荐使用端口号+ip最后一位的方式. 唯一区别ID,同一个集群 ...

  3. C++网络编程方面的开源项目

    Webbench是一个在linux下使用的非常简单的网站压测工具.它使用fork()模拟多个客户端同时访问我们设定的URL,测试网站在压力下工作的性能,最多可以模拟3万个并发连接去测试网站的负载能力. ...

  4. VO对象通过groovy模板映射XML文件

    介绍 之前写过JAVA+XSLT相关的技术博客,近期研究了一个开源工具包org.codehaus.groovy,处理VO对象和XML文件映射很方便. 简言之:将VO对象中的属性(包含Collectio ...

  5. html5中的dom中的各种节点的层次关系是怎样的

    html5中的dom中的各种节点的层次关系是怎样的 一.总结 一句话总结:Node节点是所有节点的基类,所以都继承它的方法 1.dom提供在js中动态修改html标签的作用 比如增加修改标签等,并且是 ...

  6. 编写ATL控件的简单做法

    作者:朱金灿 来源:http://blog.csdn.net/clever101 ATL并不像MFC库那样提供了很多的控件窗口类,因此要使用ATL的话需要自己去封装.封装的做法很简单.比如现在我需要一 ...

  7. 【a901】滑雪

    Time Limit: 10 second Memory Limit: 2 MB 问题描述 滑雪是一项非常刺激的运动,为了获得速度,滑雪的区域必须向下倾斜,而且当你滑到坡底,你不得不再次走上坡或者等待 ...

  8. JVM源码分析之System.currentTimeMillis及nanoTime原理详解

    JDK7和JDK8下的System.nanoTime()输出完全不一样,而且差距还非常大,是不是两个版本里的实现不一样,之前我也没注意过这个细节,觉得非常奇怪,于是自己也在本地mac机器上马上测试了一 ...

  9. 一个完整的Erlang应用

    http://blog.chinaunix.net/uid-25876834-id-3308693.html 这里介绍构建一个完整的Erlang/OTP应用的例子,最后还给出了一个在实际生成环境中,如 ...

  10. SecureCRT 向多个终端发送相同命令

    选中view,选择command windows 在下方出现的窗口中右键,接下来在窗口中输入命令即可,可以一定程度上代替分发脚本,具体请参考https://www.cnblogs.com/tele-s ...