【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.pswp.cn/bicheng/89309.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/89309.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

oracle2kingbase的字段长度问题

实验一: oracle中: create table testlen(c1 varchar2(2)); insert into testlen values(山); --成功 insert into testlen values(山西); --失败 ORA-12899: 列 "TESTK"."TESTLEN"."C1" 的值太大 (实际值: 4, 最大值: 2…

单链表的题目,咕咕咕

1.咕 203. 移除链表元素 - 力扣(LeetCode) 给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val val 的节点,并返回 新的头节点 struct ListNode* removeElements(struct ListNode* head, int val) …

关于程序=数据结构+算法这句话最近的一些思考

最近看了很多单片机STM32的的相关程序,尤其是设计到ringbuff、buffer_manage、os_memory预计mem_manage等程序中间层的用法,我对这句话有了一些更深的思考,现在记录下来,希望对处于相同阶段的程序一些思想启迪。首先“数据结构”也…

Rust 错误处理

Rust 错误处理 引言 Rust 是一种系统编程语言,以其安全、并发和性能著称。在 Rust 中,错误处理是一个核心概念,它确保了程序在遇到异常情况时能够优雅地处理。本文将深入探讨 Rust 中的错误处理机制,包括错误类型、错误传播、错误…

17. 什么是 webSocket ?

总结 WebSocket 是 HTML5 引入的一种新协议,允许客户端和服务器之间进行双向实时通信。建立在 TCP 协议之上,默认端口是 80(ws) 和 443(wss),没有同源限制,客户端可以与任意服务器通…

从零开始跑通3DGS教程:(五)3DGS训练

写在前面 本文内容 所属《从零开始跑通3DGS教程》系列文章; 本文介绍在docker中训练3dgs的方法 平台/环境 linux, nvidia GPU, docker 转载请注明出处: https://blog.csdn.net/qq_41102371/article/details/146535874 目录 写在前面系列文章准备docker创建环境参考完系列文章…

日记_7.14_实际开发的进步

1、快速定位后端2、会定位前端啦啦啦!3、前端没有意义的块叫div和span。而不是script4、所有 JavaScript 标识符均 区分大小写5、JS中$和_下划线和doller符均被视为字母。6、var、let区别:1 var全局。let局部。2 var可以重新声明格式,let之恩…

AI Agent 开发

Agent开发常用框架: LangChainLlamaIndexVercel AI SDK LangChain:一站式 LLM 应用开发框架一句话总结 LangChain 把「模型调用 外部数据 工具 记忆 流程编排」全部标准化,让你像搭积木一样快速组合出聊天机器人、RAG、Agent 等大模型应用…

【水动力学】04 二维洪水淹没模型Pypims安装

模型介绍 HiPIMS(High-Performance Integrated hydrodynamic Modelling System)使用最先进的数值方案(Godunov型有限体积法)来求解二维浅水方程以进行洪水模拟。为了支持高分辨率洪水模拟,使用CUDA/C 语言在多个GPU上…

ARC 03 从Github Action job 到 runner pod

Github Action job 分配到集群 背景 job 是 Github Action 的基本单位,每个 job 单独分配一个 runner。workflow 由一个或者多个 job 组成。如果用户触发runs-on字段为arc-runner-set的 job,那么 Github Action 服务器将 job 分配给 listener pod。 源码…

ubuntu 22.04 anaconda comfyui安装

背景: 戴尔R740服务器,安装了proxmox操作系统,配置了显卡直通。创建了一个ubuntu 22.04 VM虚拟机实例,并安装了显卡驱动与cuda等相关配置: 接下来准备搭建一套comfyui的环境,前段时间B站,抖音各…

每日面试题04:volatile字段的原理

在之前面试题02ConcurrentHashMap的底层原理中提到了volatile修饰符,在多线程编程的世界里,数据同步是一道绕不开的坎。当多个线程同时操作共享变量时,“看不见对方的修改”或“代码顺序错乱”往往会导致程序行为异常。而 volatile作为 Java …

【云原生网络】Istio基础篇

文章目录概述基础知识技术架构概述数据平面核心组件网络代理Envoy控制平面核心组件xDS协议Pilot组件其他概述参考博客😊点此到文末惊喜↩︎ 概述 基础知识 背景知识 服务网格(Service Mesh):独立于应用程序的基础设施层&#x…

PySpark Standalone 集群

一、PySpark Standalone 集群概述PySpark Standalone 集群是 Apache Spark 的一种部署模式,它不依赖于其他资源管理系统(如 YARN 或 Mesos),而是使用 Spark 自身的集群管理器。这种模式适合快速部署和测试,尤其在开发和…

图像质量评价(Image Quality Assessment,IQA)

文章目录图像质量评价(Image Quality Assessment,IQA)一、评估方式:主观评估 客观评估1.1、主观评估方式1.2、客观评估方式:全参考 半参考 无参考(1)全参考的方法对比(Full-Refer…

【跟我学YOLO】(2)YOLO12 环境配置与基本应用

欢迎关注『跟我学 YOLO』系列 【跟我学YOLO】(1)YOLO12:以注意力为中心的物体检测 【跟我学YOLO】(2)YOLO12 环境配置与基本应用 【跟我学YOLO】(3)YOLO12 用于诊断视网膜病变 【跟我学YOLO】&a…

Python爬虫实战:研究openpyxl库相关技术

1. 引言 在当今数字化时代,互联网上蕴含着海量有价值的数据。如何高效地获取这些数据并进行分析处理,成为数据科学领域的重要研究方向。网络爬虫作为一种自动化的数据采集工具,可以帮助我们从网页中提取所需的信息。而 openpyxl 作为 Python 中处理 Excel 文件的优秀库,能…

Redis学习其一

文章目录1.NoSQL概述1.1概述1.2Nosql的四大分类2.Redis入门2.1概述2.2基础知识2.2.1基础命令/语法2.2.2Redis为什么单线程还这么快2.3性能测试3.五大数据类型3.1Redis-key3.2String(字符串)3.3List(列表)3.4Set(集合)3.5Hash(哈希)3.6Zset(有…

高性能架构模式——高性能缓存架构

目录 一、引入前提二、缓存架构的设计要点2.1、缓存穿透2.1.1、缓存穿透第一种情况:存储数据不存在2.1.2、缓存穿透第二种情况:缓存数据生成耗费大量时间或者资源2.2、缓存雪崩2.2.1、解决缓存雪崩的第一种方法:更新锁机制2.2.2、解决缓存雪崩的第二种方法:后台更新机制2.3…

ubuntu+windows双系统恢复

文章目录前言一、恢复windows1.直接在grub命令行输入exit退出2.手动查找windows引导文件先ls列出所有磁盘和分区查找各个分区是否包含引导文件设置引导分区以及引导文件路径启动windows二、在windows系统下删除Ubuntu残留引导文件三、准备ubuntu系统引导盘四、安装ubuntu系统五…