C++ 实现 发送HTTP Get/Post请求

2023-05-16

1、简述

最近简单看了一下关于HTTP请求方面的知识,之前一直用Qt来实现,有专门HTTP请求的QNetworkAccessManager类来处理,实现也比较简单,这里主要讲解一下用C++代码来实现HTTP 的Get/Post请求。
一个HTTP请求报文由请求行(request line)、请求头(header)、和请求数据3个部分组成,注意请求头部分和请求数据中间需要加上“\r\n”。下图给出了请求报文的一般格式。

这里写图片描述

(1)请求行

请求行包括请求方法、URL、和HTTP协议版本三个部分。
HTTP协议的请求方法有GET、POST、HEAD、PUT、DELETE、OPTIONS、TRACE、CONNECT。这里介绍最常用的GET方法和POST方法。
GET:当客户端要从服务器中读取文档时,使用GET方法。GET方法要求服务器将URL定位的资源放在响应报文的数据部分,回送给客户端。使用GET方法时,请求参数和对应的值附加在URL后面,利用一个问号(“?”)代表URL的结尾与请求参数的开始,传递参数长度受限制。例如,/index.jsp?id=100&op=bind。
POST:当客户端给服务器提供信息较多时可以使用POST方法。POST方法将请求参数封装在HTTP请求数据中,以名称/值的形式出现,可以传输大量数据,可用来传送文件。

(2)请求头部

请求头部由关键字/值对组成,每行一对,关键字和值用英文冒号“:”分隔。请求头部通知服务器有关于客户端请求的信息,典型的请求头有:
User-Agent:产生请求的浏览器类型。
Accept:客户端可识别的内容类型列表。
Host:请求的主机名,允许多个域名同处一个IP地址,即虚拟主机。

(3)关于请求头与请求数据中间的空行

请求头之后是一个空行,需要添加 回车符和换行符——“\r\n”,通知服务器以下不再有请求头。
对于一个完整的http请求来说空行是必须的,否则服务器会认为本次请求的数据尚未完全发送到服务器,处于等待状态。

(4)请求数据

请求数据用于Post方法中。与请求数据相关的最常使用的请求头是Content-Type和Content-Length。

2、代码之路

发送Get请求

BOOL GetIpByDomainName(char *szHost, char* szIp)
{
    WSADATA        wsaData;

    HOSTENT   *pHostEnt;
    int             nAdapter = 0;
    struct       sockaddr_in   sAddr;
    if (WSAStartup(0x0101, &wsaData))
    {
        printf(" gethostbyname error for host:\n");
        return FALSE;
    }

    pHostEnt = gethostbyname(szHost);
    if (pHostEnt)
    {
        if (pHostEnt->h_addr_list[nAdapter])
        {
            memcpy(&sAddr.sin_addr.s_addr, pHostEnt->h_addr_list[nAdapter], pHostEnt->h_length);
            sprintf(szIp, "%s", inet_ntoa(sAddr.sin_addr));
        }
    }
    else
    {
        //      DWORD  dwError = GetLastError();
        //      CString  csError;
        //      csError.Format("%d", dwError);
    }
    WSACleanup();
    return TRUE;
}

void sendGetRequest()
{
     //开始进行socket初始化;
    WSADATA wData;  
    ::WSAStartup(MAKEWORD(2,2),&wData);  

    SOCKET clientSocket = socket(AF_INET,1,0);      
    struct sockaddr_in ServerAddr = {0};  
    int Ret=0;  
    int AddrLen=0;  
    HANDLE hThread=0; 

    char *bufSend = "Get /check?+参数 HTTP/1.1\r\n"  
        "Connection:Keep-Alive\r\n"
        "Accept-Encoding:gzip, deflate\r\n"  
        "Accept-Language:zh-CN,en,*\r\n"  
        "host:www.baidu.com\r\n" 
        "User-Agent:Mozilla/5.0\r\n\r\n";

    char addIp[256] = {0};
    GetIpByDomainName("www.baidu.com" , addIp);
    ServerAddr.sin_addr.s_addr = inet_addr(addIp);  
    ServerAddr.sin_port = htons(80);;  
    ServerAddr.sin_family = AF_INET;  
    char bufRecv[3069] = {0};  
    int errNo = 0;  
    errNo = connect(clientSocket,(sockaddr*)&ServerAddr,sizeof(ServerAddr));  
    if(errNo==0)  
    {  
        //如果发送成功,则返回发送成功的字节数;
        if(send(clientSocket,bufSend ,strlen(bufData),0)>0)  
        {  
            cout<<"发送成功\n";;  
        }  
        //如果接受成功,则返回接受的字节数;
         if(recv(clientSocket,bufRecv,3069,0)>0)  
         {  
            cout<<"接受的数据:"<<bufRecv<<endl;  
         }  
    }  
    else  
    {  
        errNo=WSAGetLastError();  
    }  
    //socket环境清理;
    ::WSACleanup();  
}

发送Post请求

/ post请求只需将上面的代码替换一下就可以使用
char *bufSend = "POST /check HTTP/1.1\r\n"
        "Connection:Keep-Alive\r\n"
        "Accept-Encoding:gzip, deflate\r\n"
        "Accept-Language:zh-CN,en,*\r\n"
        "Content-Length:114\r\n"
        "Content-Type:application/x-www-form-urlencoded; charset=UTF-8\r\n"
        "host:tmalarm.vemic.com\r\n"
        "User-Agent:Mozilla/5.0\r\n\r\n"
        "请求数据\r\n\r\n";

Post 请求也可以将请求数据写在请求行中,跟Get请求一样。

关于是否成功发送 Get/Post 请求

我们可以借助抓包工具看 我们发送的请求是否成功 ,可以去网上下载 HttpAnalyzerStdV7软件进行抓包,由返回的结果得出是否请求成功。
关于发送请求中 请求数据或者请求参数带 中文字符 出现乱码

我们程序中编码格式一般为Unicode编码,与HTTP服务器(UTF-8)所用编码不一样,这里就需要给中文字符转换编码格式。
关于Unicode 编码与 UTF-8编码问题 可以看一下这篇文章 C++中 Unicode 与 UTF-8 编码互转 。

char *bufSend = "Get /check?&name=%s&password=%s HTTP/1.1\r\n"  
        "Connection:Keep-Alive\r\n"
        "Accept-Encoding:gzip, deflate\r\n"  
        "Accept-Language:zh-CN,en,*\r\n"  
        "host:www.baidu.com\r\n" 
        "User-Agent:Mozilla/5.0\r\n\r\n";

CString cStrName = L"前行中的小猪";
const char* cName;
// UnicodeToUtf8方法将Unicode编码转为UTF-8格式。 
cName = UnicodeToUtf8(cStrName);
char* passWord = "123456";

char bufData[400] = {0};
sprintf_s(bufData , 400 , bufSend , cName , passWord);

//这里最终将中文转为UTF-8格式的结果保存在 bufData 数组中。
// Unicode 转 UTF-8
char* UnicodeToUtf8(const wchar_t* unicode)
{
    int len;
    len = WideCharToMultiByte(CP_UTF8, 0, unicode, -1, NULL, 0, NULL, NULL);
    char *szUtf8 = (char*)malloc(len + 1);
    memset(szUtf8, 0, len + 1);
    WideCharToMultiByte(CP_UTF8, 0, unicode, -1, szUtf8, len, NULL, NULL);
    return szUtf8;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++ 实现 发送HTTP Get/Post请求 的相关文章

  • Node.js:在检索 http 请求正文之前断开 http 请求连接

    我正在用 Node js 编写一个 http 服务器 我有一个客户端通过 HTTP POST 多部分 数据 将大文件上传到该服务器 我想接受唯一使用有效文件名上传文件的连接 我有一些条件 在服务器检索数据之前应断开无效文件名连接 我不知道如
  • Jersey 客户端异步 POST 请求不等待响应

    我创建了一个简单的 Jersey 客户端 它能够成功地使用有效负载执行 POST 请求 但现在它正在等待来自 http 端点的响应 public void callEndpoint String endpoint String payloa
  • 角度报告进度返回上传的文件总数,而不是进度

    在我的组件中我有这个 this authService addPost post subscribe data gt if data type HttpEventType UploadProgress console log data el
  • AngularJS 1.X 中的异步调用是如何工作的? $Http 调用没有返回值

    我有以下名为的函数getvalue 它与控制器一起位于 AngularJS 模块内 我试图在单击事件上调用此函数 调用控制器中的另一个函数 我希望我很清楚 功能 function getvalue Data http var value u
  • RESt api:根据身份验证对资源和内容进行识别

    我正在设计一个遵循 HATEOAS REST 原则的 API 但我不确定这个基本点 资源识别 假设这个网址 images它公开了用户 向该用户 上传的所有图像 假设我使用 oauth 访问令牌进行身份验证 images 的内容将根据授权标头
  • CURL 相当于使用 VBA 的 POST JSON 数据

    我知道这与之前提出的一些问题类似 但有些东西仍然对我不起作用 如何执行以下命令 curl X POST data statements json H Content Type application json user username p
  • Response.Redirect 并不总是重定向

    我们在一个工作不一致的页面上有一个简单的 Response Redirect IIS 6 0 大多数情况下 它会正确重定向 但我们收到一些用户抱怨 他们没有重定向 而是看到 302 对象移至此处 页面 该页面显示标题信息以及正确的位置 如果
  • 如何解析来自基于 C 的 Web 服务器的 HTTP 请求

    我有一个编程项目 我必须创建一个处理 HTTP 请求的多线程 Web 服务器 我刚刚学习了套接字编程 并且运行了一个客户端和一个服务器 我想知道解析 HTTP 请求标头的最佳方法是什么 我看到了这个 如何用C 解析http请求 https
  • Node.js 上的 Connect 出现“无法 GET /”

    我正在尝试使用以下方式开始提供一些静态网页connect像这样 var connect require connect var nowjs require now var io require socket io var app conne
  • 在 HTTP PATCH 请求中包含数据的正确方法

    当我组合 HTTP PATCH 请求时 可以选择哪些选项来包含 URL 参数之外的数据 以下任何一项都有效吗 最常见的选择是什么 多部分 表单数据 应用程序 x www form urlencoded Raw JSON 还有其他的吗 HTT
  • 下载前获取文件大小并计算已下载的文件大小 (http+ruby)

    谁能帮我 get the 之前的文件大小我开始下载 显示多少 已下载 require net http require uri url http www onalllevels com 2009 12 02TheYangShow Squid
  • 验证数据库匹配中的 $_GET id 是否足够安全?

    我的网站上有 2 个页面 一个是 index php 索引页面列出了数据库中存在的所有帖子 另一个页面是 post php 当单击索引页面上的特定帖子时 帖子页面显示单个帖子 现在我用来列出 index php 上所有帖子的代码是 post
  • Django:如何测试“HttpResponsePermanentRedirect”

    我正在为我的 django 应用程序编写一些测试 在我看来 它使用 HttpResponseRedirect 重定向到其他一些网址 那么我该如何测试呢 姜戈TestCase类有一个方法assertRedirects https docs d
  • 提交简单 PHP 表单时出现禁止错误

    我有一个不复杂的问题 这似乎比应有的更复杂 我有一个简单的表单 用于向网站添加内容 有些字段需要输入html 然而 当您在表单的不同部分输入某些 html 元素时 它会认为它讨厌您并抛出禁止的 403 错误 这是下面的表格
  • Google Closure 编译器和 multipart/form-data 不起作用

    我正在向 google 闭包编译器 API 服务发出请求 content file get contents file js url http closure compiler appspot com compile post true p
  • 处理rails应用程序中的rack_throttle异常

    当超出速率限制时 如何处理由rack throttle gem 生成的错误 现在我只收到包含以下内容的回复 Internal Server Error undefined method each for 403 Forbidden Rate
  • Unicorn + Rails + 大型上传

    我试图在使用 Rails 在 Heroku 上运行 Unicorn 时允许进行大型上传 但我意识到任何大型上传可能需要比 Unicorn 工作线程的超时时间更长的时间 这意味着 我见过这种情况发生 Unicorn 主进程将杀死上传大文件的工
  • $_POST 最大数组大小

    我有一个非常大的表单 有 gt 1000 个元素 它们已经嵌套在表单 html 结构中 foreach from result item item tr td td tr
  • 使用 python 请求在 Post 上设置文件名

    我想设置通过请求模块上传的文件的名称 files filename open myfileXXAAAZZZD rb r requests post http 127 0 0 1 5000 files files 可以使用curl 但不知道是
  • 从套接字读取 C HTTP

    我想知道如何判断是否已从套接字接收到所有数据 这是一个简单的网络代理 现在我正在处理请求部分 所以发送的内容应该以 r n r n 结尾 我不知道请求会持续多久 我在这里读过一些帖子 说我应该检查读取函数是否返回 0 但其他人说0只在客户端

随机推荐