c++ socket Get

本文展示了一个使用Winsock库发起HTTP GET请求到指定服务器的示例程序。该程序通过设置套接字参数并发送HTTP头来获取网页内容。
#ifndef UNICODE
#define UNICODE
#endif

#define WIN32_LEAN_AND_MEAN

#include "stdafx.h"
#include <stdlib.h>
#include <winsock2.h>
#include <Ws2tcpip.h>

// Link with ws2_32.lib
#pragma comment(lib, "Ws2_32.lib")

#define DEFAULT_BUFLEN 1024
#define DEFAULT_PORT 80
void myStrcat(char * desstination, int start, int size, char * source) {
    *(desstination += start);
    for (int i = 0; i < size; i++) {
        * desstination++ = * source++;
    }
}
int main() {
    system("chcp 65001");
    //----------------------
    // Declare and initialize variables.
    int iResult;
    WSADATA wsaData;

    SOCKET ConnectSocket = INVALID_SOCKET;
    struct sockaddr_in clientService;

    int recvbuflen = DEFAULT_BUFLEN;
    char * sendbuf = "GET /index.php?act=member.login HTTP/1.1\r\n"
        "Host: m.xnote.cn\r\n"
        "Connection: keep-alive\r\n\r\n";
        //"Upgrade-Insecure-Requests: 1 "
        //"User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/58.0.3029.110 Safari/537.36 "
        //"Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8 "
        //"Accept-Encoding: gzip, deflate, sdch "
        //"Accept-Language: zh-CN,zh;q=0.8 ";
    //----------------------
    // Initialize Win sock
    iResult = WSAStartup(MAKEWORD(2, 2), &wsaData);
    if (iResult != NO_ERROR) {
        wprintf(L"WSAStartup failed with error: %d ", iResult);
        system("pause");
        return 1;
    }

    //----------------------
    // Create a SOCKET for connecting to server
    ConnectSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if (ConnectSocket == INVALID_SOCKET) {
        wprintf(L"socket failed with error: %ld ", WSAGetLastError());
        WSACleanup();
        system("pause");
        return 1;
    }

    //----------------------
    // The sockaddr_in structure specifies the address family,
    // IP address, and port of the server to be connected to.
    clientService.sin_family = AF_INET;

    ADDRINFOW hints;
    ADDRINFOW * result = NULL;

    ZeroMemory(&hints, sizeof(hints));
    hints.ai_family = AF_UNSPEC;
    hints.ai_socktype = SOCK_STREAM;
    hints.ai_protocol = IPPROTO_TCP;

    iResult = GetAddrInfoW(L"m.xnote.cn", NULL, &hints, &result);
    if (iResult == -1) {
        wprintf(L"GetAddrInfoW with error: %d ", WSAGetLastError());
        closesocket(ConnectSocket);
        FreeAddrInfoW(result);
        WSACleanup();
        system("pause");
        return 1;
    }

    wchar_t ipstringbuffer[46];
    DWORD ipbufferlength = 46;

    iResult = WSAAddressToString((LPSOCKADDR)result->ai_addr, (DWORD)result->ai_addrlen, NULL, ipstringbuffer, &ipbufferlength);
    if (iResult) {
        wprintf(L"WSAAddressToString failed with %u ", WSAGetLastError());
        closesocket(ConnectSocket);
        FreeAddrInfoW(result);
        WSACleanup();
        system("pause");
        return 1;
    }

    wprintf(L"IPv4 address %ws ", ipstringbuffer);

    iResult = InetPton(AF_INET, ipstringbuffer, &clientService.sin_addr);
    if (iResult != 1) {
        wprintf(L"IP adders with error: %d ", WSAGetLastError());
        closesocket(ConnectSocket);
        FreeAddrInfoW(result);
        WSACleanup();
        system("pause");
        return 1;
    }

    clientService.sin_port = htons(DEFAULT_PORT);

    //----------------------
    // Connect to server.
    iResult = connect(ConnectSocket, (SOCKADDR * )&clientService, sizeof(clientService));
    if (iResult == SOCKET_ERROR) {
        wprintf(L"connect failed with error: %d ", WSAGetLastError());
        closesocket(ConnectSocket);
        FreeAddrInfoW(result);
        WSACleanup();
        system("pause");
        return 1;
    }

    //----------------------
    // Send an initial buffer
    iResult = send(ConnectSocket, sendbuf, (int)strlen(sendbuf), 0);
    if (iResult == SOCKET_ERROR) {
        wprintf(L"send failed with error: %d ", WSAGetLastError());
        closesocket(ConnectSocket);
        FreeAddrInfoW(result);
        WSACleanup();
        system("pause");
        return 1;
    }

    printf("Bytes Sent: %d \n", iResult);

    char * recvbuf = (char *)malloc(DEFAULT_BUFLEN * sizeof(char));
    char * printbuf = (char * )malloc(DEFAULT_BUFLEN * sizeof(char));
    memset(printbuf, 0, sizeof(printbuf));
    int printbufSize = 0;
    int time_out = 2000; // 2秒
    setsockopt(ConnectSocket, SOL_SOCKET, SO_RCVTIMEO, (char *)&time_out, sizeof(time_out));
    // Receive until the peer closes the connection
    do {

        iResult = recv(ConnectSocket, recvbuf, recvbuflen, 0);
        if (iResult > 0)
        {
            wprintf(L"Bytes received: %d \n", iResult);
            printbufSize += iResult;
            printbuf = (char *)realloc(printbuf, printbufSize);
            myStrcat(printbuf, printbufSize - iResult, iResult, recvbuf);
        }
        else if (iResult == 0)
            wprintf(L"Connection closed \n");
        else
            wprintf(L"recv failed with error: %d \n", WSAGetLastError());

    } while (iResult > 0);
    *(printbuf + printbufSize) = '\0';
    wprintf(L"%S\n", printbuf);
    // close the socket
    iResult = closesocket(ConnectSocket);
    if (iResult == SOCKET_ERROR) {
        wprintf(L"close failed with error: %d ", WSAGetLastError());
        FreeAddrInfoW(result);
        WSACleanup();
        system("pause");
        return 1;
    }

    FreeAddrInfoW(result);
    WSACleanup();
    return 0;
}

这里写图片描述

<think> 我们参考用户的查询和提供的引用内容,用户需要的是使用C++通过socket通讯实现HTTP GET请求的方法。 根据引用[1]和引用[2],我们知道HTTP是基于TCP的,因此我们需要建立TCP连接(使用Socket),然后按照HTTP协议发送请求。 引用[3]提供了一个函数调用的示例,展示了如何调用一个名为SOCK_HTTP_GET的函数来发送GET请求。 步骤分解: 1. 建立Socket连接(使用TCP协议)。 2. 准备HTTP GET请求报文。 3. 通过Socket发送请求报文。 4. 接收服务器的响应。 5. 关闭Socket连接。 我们将编写一个函数来实现这个过程,函数名为SOCK_HTTP_GET(如引用[3]所示),参数包括请求的路径(包含查询参数)、服务器地址(域名或IP)、端口号。 注意:HTTP GET请求报文的格式通常如下: ``` GET /path?query HTTP/1.1 Host: example.com [其他头部字段] \r\n\r\n ``` 我们这里为了简单,可以只发送必要的部分:GET行和Host头,然后两个换行表示结束。 由于引用[3]的示例中第一个参数是完整的路径和查询字符串(例如:"/api/Materiel/TestGet?id=123&name=46798"),第二个参数是服务器地址(不带协议),第三个参数是端口。 实现细节: - 创建socket,使用IPv4(AF_INET),流式传输(SOCK_STREAM),协议为TCP(0)。 - 解析服务器地址:如果给的是域名,需要转换成IP地址(使用gethostbyname或getaddrinfo)。 - 连接服务器(connect)。 - 构造请求字符串。 - 发送请求(send)。 - 接收响应(recv),直到接收完所有数据。 - 关闭socket。 注意:我们不考虑HTTPS(因为题目要求是HTTP),并且假设服务器是80端口(默认HTTP端口)。 下面我们给出代码实现(适用于Linux/macOS,Windows下需要使用Winsock,但原理类似): </think> ### 使用C++通过Socket实现HTTP GET请求的方法 以下是基于Socket实现HTTP GET请求的完整步骤和代码实现(适用于Linux/macOS系统): #### 核心步骤 1. **建立Socket连接**:创建TCP套接字并连接服务器 2. **构造HTTP请求**:按照RFC标准格式组装GET请求 3. **发送请求**:通过Socket发送HTTP报文 4. **接收响应**:读取服务器返回的HTTP响应 5. **解析处理**:提取响应头和响应体 #### 代码实现 ```cpp #include <iostream> #include <string> #include <sys/socket.h> #include <arpa/inet.h> #include <unistd.h> #include <netdb.h> const int BUFFER_SIZE = 4096; void HTTP_GET(const std::string& path, const std::string& host, int port) { // 1. 创建Socket int sock = socket(AF_INET, SOCK_STREAM, 0); if (sock < 0) { perror("Socket creation failed"); return; } // 2. 解析主机地址 struct hostent* server = gethostbyname(host.c_str()); if (!server) { std::cerr << "ERROR: Host not found\n"; close(sock); return; } // 3. 设置服务器地址结构 struct sockaddr_in serv_addr; memset(&serv_addr, 0, sizeof(serv_addr)); serv_addr.sin_family = AF_INET; memcpy(&serv_addr.sin_addr.s_addr, server->h_addr, server->h_length); serv_addr.sin_port = htons(port); // 4. 连接服务器 if (connect(sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr)) < 0) { perror("Connection failed"); close(sock); return; } // 5. 构造HTTP请求 std::string request = "GET " + path + " HTTP/1.1\r\n" "Host: " + host + "\r\n" "Connection: close\r\n\r\n"; // 非持久连接 // 6. 发送请求 if (send(sock, request.c_str(), request.length(), 0) < 0) { perror("Request send failed"); close(sock); return; } // 7. 接收响应 char buffer[BUFFER_SIZE]; std::string response; int bytes_received; while ((bytes_received = recv(sock, buffer, BUFFER_SIZE - 1, 0)) > 0) { buffer[bytes_received] = '\0'; response += buffer; } // 8. 输出响应 std::cout << "=== HTTP Response ===" << std::endl; std::cout << response << std::endl; // 9. 关闭Socket close(sock); } int main() { // 示例调用(请求百度首页) HTTP_GET("/", "www.baidu.com", 80); return 0; } ``` #### 关键说明 1. **请求格式要求**: - GET请求行格式:`GET /path?query HTTP/1.1` - 必须包含Host头:`Host: example.com` - 需要两个CRLF(`\r\n\r\n`)结束头部[^2][^3] 2. **连接处理**: - 使用`gethostbyname()`解析域名 - 端口80是HTTP默认端口 - `Connection: close`声明非持久连接 3. **错误处理**: - 检查所有Socket操作的返回值 - 资源使用后立即释放(`close(sock)`) #### 编译运行 ```bash # Linux/macOS编译命令 g++ -o http_get http_get.cpp ./http_get ``` #### 注意事项 1. Windows系统需使用Winsock库,初始化流程不同 2. 实际应用需添加超时处理机制 3. 对于HTTPS需要SSL/TLS加密层(使用OpenSSL等库) 4. 需处理分块传输编码(Transfer-Encoding: chunked)[^2]
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值