当前位置: 首页 > news >正文

【C语言网络编程】HTTP 客户端请求(发送请求报文过程)

在 C 语言中,我们可以使用 socket 编程来手动实现一个简单的 HTTP 客户端,像浏览器一样请求网页数据。本文将结合实际代码,重点讲解如何通过 C 语言构造并发送一个 HTTP 请求报文,实现与服务器的基本通信。

文章目标

通过一个简单的 http_send_request() 函数,我们将实现以下流程:

  1. 将域名(如 "www.baidu.com")解析成 IP 地址

  2. 与目标服务器建立 TCP 连接(80 端口)

  3. 构造 HTTP 请求报文并发送给服务器

一、代码结构总览

#define HTTP_VERSION        "HTTP/1.1"
#define CONNETION_TYPE      "Connection:close\r\n"
#define BUFFER_SIZE         4096

我们使用 HTTP/1.1 协议,连接类型为短连接(发送请求后关闭)。

二、域名解析函数:host_to_ip

char *host_to_ip(const char *hostname) {struct hostent *host_entry = gethostbyname(hostname);  // DNS 查询if (host_entry) {return inet_ntoa(*(struct in_addr*)host_entry->h_addr_list[0]);  // 返回IP字符串}return NULL;
}
  • gethostbyname() 负责 DNS 解析

  • inet_ntoa() 将原始 IP 地址(二进制)转换为点分十进制字符串,如 "14.215.177.39"

三、创建并连接 Socket:http_create_socket

int http_create_socket(char *ip) {int sockfd = socket(AF_INET, SOCK_STREAM, 0);  // 创建 TCP socketstruct sockaddr_in sin = {0};sin.sin_family = AF_INET;sin.sin_port = htons(80);                     // 设置端口:HTTP 默认 80sin.sin_addr.s_addr = inet_addr(ip);          // 将 IP 字符串转换为网络地址if (0 != connect(sockfd, (struct sockaddr*)&sin, sizeof(sin))) {return -1;  // 连接失败}fcntl(sockfd, F_SETFL, O_NONBLOCK);           // 设置非阻塞模式(可选)return sockfd;
}

四、发送 HTTP 请求:http_send_request

这是本文的重点,完整代码如下:

char * http_send_request(const char *hostname, const char *resource) {char *ip = host_to_ip(hostname);               // 1. 域名转 IPint sockfd = http_create_socket(ip);           // 2. 创建 TCP 连接char buffer[BUFFER_SIZE] = {0};                // 3. 准备请求报文缓冲区// 4. 构造 HTTP GET 请求报文sprintf(buffer,"GET %s %s\r\n""Host: %s\r\n""%s\r\n",resource, HTTP_VERSION, hostname, CONNETION_TYPE);// 5. 发送请求数据send(sockfd, buffer, strlen(buffer), 0);return NULL;  // 当前版本未实现接收部分
}

五、HTTP 报文解析说明

通过 sprintf() 构造的请求报文如下所示(举例):

GET /index.html HTTP/1.1
Host: www.baidu.com
Connection: close

它由以下部分组成:

行数内容说明
第1行请求行指定方法、资源路径、协议版本
第2行Host 头告诉服务器你访问的是哪个域名
第3行Connection 头表示用完连接后立即关闭
空行必须表示请求头结束,开始正文(此处没有正文)

\r\n 是 HTTP 标准要求的换行符,不能用 \n 替代。

六、http_send_request() 函数流程图

开始││ 输入参数:hostname 和 resource│├─▶ 1. 通过 host_to_ip(hostname)│     └─ DNS 查询 → 获取 IP 地址(如 "14.215.177.39")│├─▶ 2. 调用 http_create_socket(ip)│     └─ 创建 TCP socket 并连接服务器 80 端口│├─▶ 3. 构造 HTTP 请求报文│     └─ 格式如下:│         GET /resource HTTP/1.1│         Host: hostname│         Connection: close│├─▶ 4. 使用 send() 发送请求数据到 socket│└─▶ 5. 当前版本未实现 recv(),结束函数

域名 → IP → TCP连接 → 构造请求 → 发送数据

七、完整代码

#define HTTP_VERSION        "HTTP/1.1"              // 指定使用的 HTTP 协议版本
#define CONNETION_TYPE      "Connection:close\r\n"  // 设置连接类型为关闭连接(短连接)#define BUFFER_SIZE 4096                             // 定义请求缓冲区大小// 将主机名(域名)转换为 IP 地址字符串
char *host_to_ip(const char *hostname) {struct hostent *host_entry = gethostbyname(hostname);   // 调用 DNS 查询函数// 如果查询成功,返回对应 IP 地址(点分十进制字符串)// h_addr_list 是 IP 地址列表,取第一个并转换为字符串if (host_entry) {return inet_ntoa((struct in_addr*)*host_entry->h_addr_list);}// 查询失败返回 NULLreturn NULL;
}// 创建一个 TCP socket 并连接到指定 IP 地址的 80 端口
int http_create_socket(char *ip) {int sockfd = socket(AF_INET, SOCK_STREAM, 0);     // 创建 TCP socketstruct sockaddr_in sin = {0};                     // 初始化服务器地址结构sin.sin_family = AF_INET;                         // 使用 IPv4 协议sin.sin_port = htons(80);                         // 设置端口为 80,使用 htons 转换为网络字节序sin.sin_addr.s_addr = inet_addr(ip);              // 将 IP 字符串转换为网络字节序// 尝试连接服务器if (0 != connect(sockfd, (struct sockaddr*)&sin, sizeof(struct sockaddr_in))) {return -1;                                     // 连接失败则返回 -1}fcntl(sockfd, F_SETFL, O_NONBLOCK);               // 设置 socket 为非阻塞模式(可选)return sockfd;                                    // 返回连接成功的 socket 文件描述符
}// 构造并发送一个 HTTP GET 请求
char * http_send_request(const char *hostname, const char *resource) {char *ip = host_to_ip(hostname);                  // 第一步:通过域名获取 IP 地址int sockfd = http_create_socket(ip);              // 第二步:创建并连接 socket 到服务器char buffer[BUFFER_SIZE] = {0};                   // 初始化发送缓冲区// 第三步:构造 HTTP 请求报文// 组成部分包括请求行、Host 头部、Connection 头部sprintf(buffer,"GET %s %s\r\n"           // 请求行:GET /path HTTP/1.1"Host: %s\r\n"            // Host 头:指定服务器域名"%s\r\n",                 // Connection: close(关闭连接)resource, HTTP_VERSION,hostname,CONNETION_TYPE);// 第四步:通过 socket 发送请求报文send(sockfd, buffer, strlen(buffer), 0);return NULL; // 当前函数版本没有实现响应接收,暂时返回 NULL
}

https://github.com/0voice

http://www.lryc.cn/news/590163.html

相关文章:

  • Mybatis07-缓存
  • 比特币技术简史 第二章:密码学基础 - 哈希函数、公钥密码学与数字签名
  • 今日行情明日机会——20250716
  • S7-200 SMART PLC:模拟量模块接线全解析
  • 汽车功能安全-相关项集成和测试(系统集成测试系统合格性测试)-12
  • xss-labs通关
  • “Datawhale AI夏令营”基于带货视频评论的用户洞察挑战赛2
  • lesson15:Python的文件操作
  • Docker 中的动态配置:docker update 命令与环境变量管理
  • Hadoop架构演进:从1.0到2.0的深度对比与优化解析
  • Docker 安装和配置 MySQL 8.0.36 的详细步骤
  • 力扣-146.LRU缓存机制
  • Linux-局域网构建+VLAN 划分 + 端口 MAC-IP 绑定 + 静态 DHCP
  • 【前端】在Vue3中绘制多系列柱状图与曲线图
  • (nice!!!)(LeetCode 每日一题) 3201. 找出有效子序列的最大长度 I (动态规划dp)
  • 产品经理笔试考试回忆集(2025湖南某国企)
  • 电力政策解读:山东电网新型储能集中调用的能源管理系统实现点
  • 百炼Agent MCP与IoT实战(二):阿里云MQTT Broker配置
  • arm版本的ubuntu安装git或者vim等方法
  • TypeScript的export用法
  • Linux LVS集群技术详解与实战指南
  • Vue + React 联合开发指南:跨越框架边界的前端实践
  • 第二章【vue】基础(超详细)
  • 佰力博检测与您探讨高温压电d33测试的操作步骤与选购建议
  • go项目实战
  • 自学中医笔记(一)
  • PowerBI实现仅在需要图表时显示图表
  • 时序大模型为时序数据库带来的变革与机遇
  • 从零开始的云计算生活——番外3,LVS+KeepAlived+Nginx高可用实现方案
  • AWS权限异常实时告警系统完整实现指南