《TCP/IP网络编程》学习笔记 | Chapter 24:制作 HTTP 服务器端

《TCP/IP网络编程》学习笔记 | Chapter 24:制作 HTTP 服务器端

  • 《TCP/IP网络编程》学习笔记 | Chapter 24:制作 HTTP 服务器端
    • HTTP 概要
      • 理解 Web 服务器端
      • 无状态的 Stateless 协议
      • 请求消息(Request Message)的结构
      • 响应消息(Response Message)的结构
    • 实现简单的 Web 服务器端
      • 基于 Windows 的多线程 Web 服务器端
      • 基于 Linux 的多线程 Web 服务器端
    • 习题
      • (1)下列关于 Web 服务器端和 Web 浏览器端的说法错误的是?
      • (2)下列关于 HTTP 协议的描述错误的是?
      • (3)IOCP 和 epoll 是可以保证高性能的典型服务器端模型,但如果在基于 HTTP 协议的 Web 服务器端使用这些模型,则无法保证一定能得到高性能。请说明原因。

《TCP/IP网络编程》学习笔记 | Chapter 24:制作 HTTP 服务器端

HTTP 概要

本章将编写 HTTP(HyperText Transfer Protocol,超文本传输协议)服务器端,即 Web 服务器端。

理解 Web 服务器端

HTTP 是以超文本传输为目的而设计的应用层协议,基于 TCP/IP 实现。

Web 服务器端就是基于 HTTP 协议,将网页对应文件传输给客户端的服务器端。

无状态的 Stateless 协议

HTTP 的请求及相应方式:

在这里插入图片描述

从上图可以看出,服务器端相应客户端请求后立即断开连接。换言之,服务器端不会维持客户端状态。即使同一客户端再次发送请求,服务器端也无法辨认出是原先那个,而会以相同方式处理新请求。因此,HTTP 又称「无状态的 Stateless 协议」。

现代互联网为了解决这种缺陷,会使用 Cookie、Session 等保存客户端信息。详见于:https://blog.csdn.net/ProgramNovice/article/details/137809102。

请求消息(Request Message)的结构

下面是客户端向服务端发起请求消息的结构:

在这里插入图片描述

从图中可以看出,请求消息可以分为请求头、消息头、消息体 3 个部分。

其中,请求行含有请求方式(请求目的)信息。典型的请求方式有 GET 和 POST ,GET 主要用于请求数据,POST 主要用于传输数据。

为了降低复杂度,我们实现只能响应 GET 请求的 Web 服务器端。

「GET/index.html HTTP/1.1」 具有如下含义:请求(GET)index.html 文件,通常以 1.1 版本的 HTTP 协议进行通信。

请求行下面的消息头中包含发送请求的浏览器信息、用户认证信息等关于 HTTP 消息的附加信息。最后的消息体中装有客户端向服务端传输的数据,为了装入数据,需要以 POST 方式发送请求。但是我们的目标是实现 GET 方式的服务器端,所以可以忽略这部分内容。另外,消息体和消息头与之间以空行隔开,因此不会发生边界问题。

响应消息(Response Message)的结构

下面是 Web 服务器端向客户端传递的响应信息的结构。从图中可以看出,该响应消息由状态行、头信息、消息体等 3 个部分组成。

状态行中有关于请求的状态信息,这是与请求消息相比最为显著的区别。

在这里插入图片描述

第一个字符串状态行中含有关于客户端请求的处理结果。例如,客户端请求 index.html 文件时,表示 index.html 文件是否存在、服务端是否发生问题而无法响应等不同情况的信息写入状态行。图中的「HTTP/1.1 200 OK」具有如下含义:我想以 1.1 版本的 HTTP 协议进行响应,你的请求已正确处理。

这里的数字叫做状态码,具体可以通过下面的文章详细了解:

  1. https://blog.csdn.net/ProgramNovice/article/details/136882012
  2. https://blog.csdn.net/ProgramNovice/article/details/126164376

消息头中含有传输的数据类型和长度等信息。图中的消息头含有如下信息:服务端名为 SimpleWebServer ,传输的数据类型为 text/html。数据长度不超过 2048 个字节。

最后插入一个空行后,通过消息体发送客户端请求的文件数据。

以上就是实现 Web 服务端过程中必要的 HTTP 协议。

实现简单的 Web 服务器端

基于 Windows 的多线程 Web 服务器端

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <winsock2.h>
#include <process.h>#define BUF_SIZE 2048
#define BUF_SMALL 100unsigned WINAPI RequestHandler(void *arg);
void SendData(SOCKET sock, char *ct, char *fileName);
void SendErrorMSG(SOCKET sock);
char *ContentType(char *file);
void ErrorHanding(char *message);int main(int argc, char *argv[])
{WSADATA wsaData;SOCKET hServSock, hClntSock;SOCKADDR_IN servAddr, clntAddr;HANDLE hThread;DWORD dwThreadID;int clntAdrSize;if (argc != 2){printf("Usage : %s <port>\n", argv[0]);exit(1);}if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)ErrorHanding("WSAStartup() error!");hServSock = socket(PF_INET, SOCK_STREAM, 0);if (hServSock == INVALID_SOCKET)ErrorHanding("socket() error!");memset(&servAddr, 0, sizeof(servAddr));servAddr.sin_family = AF_INET;servAddr.sin_addr.s_addr = htonl(INADDR_ANY);servAddr.sin_port = htons(atoi(argv[1]));if (bind(hServSock, (SOCKADDR *)&servAddr, sizeof(servAddr)) == SOCKET_ERROR)ErrorHanding("bind() error!");if (listen(hServSock, 5) == SOCKET_ERROR)ErrorHanding("listen() error!");while (1){clntAdrSize = sizeof(clntAddr);hClntSock = accept(hServSock, (SOCKADDR *)&clntAddr, &clntAdrSize);if (hClntSock == INVALID_SOCKET)ErrorHanding("accept() error!");printf("Connection Request: %s:%d\n", inet_ntoa(clntAddr.sin_addr), ntohs(clntAddr.sin_port));hThread = (HANDLE)_beginthreadex(NULL, 0, RequestHandler, (void *)hClntSock, 0, (unsigned *)&dwThreadID);}closesocket(hServSock);WSACleanup();return 0;
}unsigned WINAPI RequestHandler(void *arg)
{SOCKET hClntSock = *(SOCKET *)arg;char buf[BUF_SIZE];char method[BUF_SMALL];char ct[BUF_SMALL];char fileName[BUF_SMALL];recv(hClntSock, buf, BUF_SIZE, 0);if (strstr(buf, "HTTP/") == NULL) // 查看是否为 HTTP 提出的请求{SendErrorMSG(hClntSock);closesocket(hClntSock);return 1;}strcpy(method, strtok(buf, " /"));if (strcmp(method, "GET")) // 查看是否为 GET 请求{SendErrorMSG(hClntSock);}strcpy(fileName, strtok(NULL, " /")); // 查看请求文件名strcpy(ct, ContentType(fileName));    // 查看 Content-TypeSendData(hClntSock, ct, fileName);return 0;
}// 读取文件内容,作为响应数据发送
void SendData(SOCKET sock, char *ct, char *fileName)
{char protocol[] = "HTTP/1.0 200 OK\r\n";char servName[] = "Server:simple web server\r\n";char cntLen[] = "Content-Length:2048\r\n";char cntType[BUF_SMALL];char buf[BUF_SIZE];sprintf(cntType, "Content-Type:%s\r\n\r\n", ct);FILE *fp;if ((fp = fopen(fileName, "r")) == NULL){SendErrorMSG(sock);return;}// 传输头信息send(sock, protocol, strlen(protocol), 0);send(sock, servName, strlen(servName), 0);send(sock, cntLen, strlen(cntLen), 0);send(sock, cntType, strlen(cntType), 0);// 传输请求数据while (fgets(buf, BUF_SIZE, fp) != NULL){send(sock, buf, strlen(buf), 0);}closesocket(sock);
}// 返回错误响应
void SendErrorMSG(SOCKET sock)
{char protocol[] = "HTTP/1.0 400 Bad Request\r\n";char servName[] = "Server:simple web server\r\n";char cntLen[] = "Content-Length:2048\r\n";char cntType[] = "Content-Type:text/html\r\n\r\n";char content[] = "<html><head><title>NETWORK</title></head>""<body><font size=+5><br>发生错误!查看请求文件名和请求方式!</font>""</body></html>";send(sock, protocol, strlen(protocol), 0);send(sock, servName, strlen(servName), 0);send(sock, cntLen, strlen(cntLen), 0);send(sock, cntType, strlen(cntType), 0);send(sock, content, strlen(content), 0);closesocket(sock);
}// 根据文件名的后缀返回对应的 MIME 类型
char *ContentType(char *file)
{char extension[BUF_SMALL];char fileName[BUF_SMALL];strcpy(fileName, file);strtok(fileName, ".");                // 取出文件名strcpy(extension, strtok(NULL, ".")); // 取出文件后缀名if (!strcmp(extension, "html") || !strcmp(extension, "htm"))return "text/html";elsereturn "text/plain";
}void ErrorHanding(char *message)
{fputs(message, stderr);fputc('\n', stderr);exit(1);
}

编译:

gcc webserver_win.c -lws2_32 -o webserv_win

运行结果:

在这里插入图片描述

基于 Linux 的多线程 Web 服务器端

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <pthread.h>#define BUF_SIZE 1024
#define SMALL_BUF 100void *request_handler(void *arg);
void send_data(FILE *fp, char *ct, char *file_name);
char *content_type(char *file);
void send_error(FILE *fp);
void error_handling(char *message);int main(int argc, char *argv[])
{int serv_sock, clnt_sock;struct sockaddr_in serv_adr, clnt_adr;int clnt_adr_size;char buf[BUF_SIZE];pthread_t t_id;if (argc != 2){printf("Usage : %s <port>\n", argv[0]);exit(1);}serv_sock = socket(PF_INET, SOCK_STREAM, 0);memset(&serv_adr, 0, sizeof(serv_adr));serv_adr.sin_family = AF_INET;serv_adr.sin_addr.s_addr = htonl(INADDR_ANY);serv_adr.sin_port = htons(atoi(argv[1]));if (bind(serv_sock, (struct sockaddr *)&serv_adr, sizeof(serv_adr)) == -1)error_handling("bind() error");if (listen(serv_sock, 20) == -1)error_handling("listen() error");while (1){clnt_adr_size = sizeof(clnt_adr);clnt_sock = accept(serv_sock, (struct sockaddr *)&clnt_adr, &clnt_adr_size);printf("Connection Request : %s:%d\n",inet_ntoa(clnt_adr.sin_addr), ntohs(clnt_adr.sin_port));pthread_create(&t_id, NULL, request_handler, &clnt_sock);pthread_detach(t_id);}close(serv_sock);return 0;
}void *request_handler(void *arg)
{int clnt_sock = *((int *)arg);char req_line[SMALL_BUF];FILE *clnt_read;FILE *clnt_write;char method[10];char ct[15];char file_name[30];clnt_read = fdopen(clnt_sock, "r");clnt_write = fdopen(dup(clnt_sock), "w");fgets(req_line, SMALL_BUF, clnt_read);if (strstr(req_line, "HTTP/") == NULL){send_error(clnt_write);fclose(clnt_read);fclose(clnt_write);return;}strcpy(method, strtok(req_line, " /"));strcpy(file_name, strtok(NULL, " /"));strcpy(ct, content_type(file_name));if (strcmp(method, "GET") != 0){send_error(clnt_write);fclose(clnt_read);fclose(clnt_write);return;}fclose(clnt_read);send_data(clnt_write, ct, file_name);
}
void send_data(FILE *fp, char *ct, char *file_name)
{char protocol[] = "HTTP/1.0 200 OK\r\n";char server[] = "Server:Linux Web Server \r\n";char cnt_len[] = "Content-length:2048\r\n";char cnt_type[SMALL_BUF];char buf[BUF_SIZE];FILE *send_file;sprintf(cnt_type, "Content-type:%s\r\n\r\n", ct);send_file = fopen(file_name, "r");if (send_file == NULL){send_error(fp);return;}// 传输头信息fputs(protocol, fp);fputs(server, fp);fputs(cnt_len, fp);fputs(cnt_type, fp);// 传输请求数据while (fgets(buf, BUF_SIZE, send_file) != NULL){fputs(buf, fp);fflush(fp);}fflush(fp);fclose(fp);
}
char *content_type(char *file)
{char extension[SMALL_BUF];char file_name[SMALL_BUF];strcpy(file_name, file);strtok(file_name, ".");strcpy(extension, strtok(NULL, "."));if (!strcmp(extension, "html") || !strcmp(extension, "htm"))return "text/html";elsereturn "text/plain";
}
void send_error(FILE *fp)
{char protocol[] = "HTTP/1.0 400 Bad Request\r\n";char server[] = "Server:Linux Web Server \r\n";char cnt_len[] = "Content-length:2048\r\n";char cnt_type[] = "Content-type:text/html\r\n\r\n";char content[] = "<html><head><title>NETWORK</title></head>""<body><font size=+5><br>发生错误! 查看请求文件名和请求方式!""</font></body></html>";fputs(protocol, fp);fputs(server, fp);fputs(cnt_len, fp);fputs(cnt_type, fp);fflush(fp);
}
void error_handling(char *message)
{fputs(message, stderr);fputc('\n', stderr);exit(1);
}

编译:

gcc webserver_linux.c -D_REENTRANT -o webserver_linux -lpthread

运行结果:

在这里插入图片描述

习题

(1)下列关于 Web 服务器端和 Web 浏览器端的说法错误的是?

a. Web 浏览器并不是通过自身创建的套接字连接服务端的客户端。
b. Web 服务器端通过 TCP 套接字提供服务,因为它将保持较长的客户端连接并交换数据。
c. 超文本与普通文本的最大区别是其具有可跳转的特性。
d. Web 服务器端可视为向浏览器提供请求文件的文件传输服务器端。
e. 除 Web 浏览器外,其他客户端都无法访问 Web 服务器端。

答:

a、b、e。

(2)下列关于 HTTP 协议的描述错误的是?

a. HTTP 协议是无状态的 Stateless 协议,不仅可以通过 TCP 实现,还可以通过 UDP 来实现
b. HTTP 协议是无状态的 Stateless 协议,因为其在 1 次请求和响应过程完成后立即断开连接。因此,如果同一服务器端和客户端需要 3 次请求及响应,则意味着需要经过 3 次套接字的创建过程。
c. 服务端向客户端传递的状态码中含有请求处理结果的信息。
d. HTTP 协议是基于因特网的协议,因此,为了同时向大量客户端提供服务,HTTP 协议被设计为 Stateless 协议。

答:

a。

(3)IOCP 和 epoll 是可以保证高性能的典型服务器端模型,但如果在基于 HTTP 协议的 Web 服务器端使用这些模型,则无法保证一定能得到高性能。请说明原因。

IOCP 和 epoll 解决了 网络 I/O 的高效调度,但 Web 服务器的整体性能受制于:

  1. 应用层逻辑的复杂度。
  2. HTTP 协议的固有特性,比如:客户端和服务器端交换 1 次数据后将立即断开连接,没有足够的时间发挥 IOCP 或 epoll 的优势。
  3. 系统资源管理,如线程池、内存分配。
  4. 外部依赖的性能,如数据库、网络带宽。

因此,单纯依赖高性能 I/O 模型无法保证 Web 服务器的高性能,需结合协议优化(如 HTTP/2、QUIC)、代码逻辑优化、分布式架构设计(如负载均衡、缓存)等多方面改进。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/77149.shtml

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

相关文章

【Quest开发】在虚拟世界设置具有遮挡关系的透视窗口

软件&#xff1a;Unity 2022.3.51f1c1、vscode、Meta XR All in One SDK V72 硬件&#xff1a;Meta Quest3 仅针对urp管线 参考了YY老师这篇&#xff0c;可以先看他的再看这个可能更好理解一些&#xff1a;Unity Meta Quest MR 开发&#xff08;七&#xff09;&#xff1a;使…

GPU 招投标全流程分析与总结

GPU 招投标全流程分析与总结 招投标流程概述 以下是通过代理商采购Nvidia H20-GPU 141G的招投标全流程分析: #mermaid-svg-hMPPfkCpGj8GKXfV {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-hMPPfkCpGj8GKXfV .er…

[C++] STL中的向量容器<vector>附加练习

目录 讲在前面(必看)八卦阵题目描述输入格式输出格式输入输出样例数据范围AC代码及要点 决赛应援题目描述输入格式输出格式输入输出样例数据范围AC代码及要点 讲在前面(必看) 本篇为练习篇, vector讲解篇在这里. 菜鸟食用前请做好心理准备(你懂的) 八卦阵 题目描述 n 名同学…

基于SpringBoot+Vue3实现的宠物领养管理平台功能一

一、前言介绍&#xff1a; 1.1 项目摘要 随着社会经济的发展和人们生活水平的提高&#xff0c;越来越多的人开始关注并参与到宠物领养中。宠物已经成为许多家庭的重要成员&#xff0c;人们对于宠物的关爱和照顾也日益增加。然而&#xff0c;传统的宠物领养流程存在诸多不便&a…

parameter和localparam的区别(verilog中)

在Verilog中&#xff0c;parameter 和 localparam 都用于定义常量&#xff0c;但是它们之间有一些重要的区 作用范围&#xff1a; parameter&#xff1a;可以在模块外部被修改或重定义。它可以被作为模块的参数传递给其他模块&#xff0c;因此具有较广泛的作用范围&#xff0c;…

鸿蒙API15 “一多开发”适配:解锁黄金三角法则,开启高效开发新旅程

一、引言 在万物互联的时代浪潮中&#xff0c;鸿蒙操作系统以其独特的 “一多开发” 理念&#xff0c;为开发者打开了一扇通往全场景应用开发的新大门。“一多开发”&#xff0c;即一次开发&#xff0c;多端部署 &#xff0c;旨在让开发者通过一套代码工程&#xff0c;就能高效…

Linux中docker容器拉取镜像失败解决方案

查看 /etc/systemd/system/docker.service.d/http-proxy.conf 文件&#xff08;没有则新建&#xff09;&#xff0c;查看自定义 Docker 服务的代理设置 输入内容 [Service] Environment"HTTP_PROXYsocks5://10.211.13.214:7890" Environment"HTTPS_PROXYsocks…

半导体设备通信标准—secsgem v0.3.0版本使用说明文档(2)之GEM(SEMI 30)

文章目录 1、处理器1.1、事件 2、GEM 合规性2.1、状态模型2.2、 设备加工状态2.3、 文档2.4、 控制 &#xff08;作员启动&#xff09;2.5、 动态事件报告配置2.6、 跟踪数据收集2.7、 报警管理2.8、 远程控制2.9、 设备常量2.10、 工艺配方管理2.11、 物料移动2.12、 设备终端…

每日算法-链表(23.合并k个升序链表、25.k个一组翻转链表)

一.合并k个升序链表 1.1题目描述 1.2题解思路 解法一&#xff1a;小根堆 我们可以先定义一个小根堆&#xff0c;将k个指针的头结点如堆&#xff0c;每次取堆顶元素尾插到newhead中&#xff0c;然后再pop()&#xff0c;接着push堆顶原来堆顶元素的下一个节点 重点分析&#…

Java性能剖析工具箱

1. 基础知识 1.1 Java性能调优概述 1.1.1 性能调优的重要性 性能调优是提升系统效率、降低成本和增强用户体验的关键步骤。通过优化,可以减少响应时间、降低资源消耗并提高系统的稳定性和可扩展性。 1.1.2 性能问题的常见表现 高CPU使用率:可能由热点方法或线程阻塞引起。…

如何使用SpringApplicationRunListener在Spring Boot 应用的不同生命周期阶段插入自定义逻辑

目录 一、引言二、核心方法概述三、加载机制四、使用场景五、扩展 - 如何在测试的不同阶段插入逻辑5.1 TestExecutionListener & AbstractTestExecutionListener5.1.1 主要功能5.1.2 生命周期方法 5.2 如何集成TestExecutionListener5.3 总结 一、引言 SpringApplicationR…

【NLP】 19. Tokenlisation 分词 BPE, WordPiece, Unigram/SentencePiece

1. 翻译系统性能评价方法 在机器翻译系统性能评估中&#xff0c;通常既有人工评价也有自动评价方法&#xff1a; 1.1 人工评价 人工评价主要关注以下几点&#xff1a; 流利度&#xff08;Fluency&#xff09;&#xff1a; 判断翻译结果是否符合目标语言的语法和习惯。充分性…

openai发布今天发布了o3和o4-mini。

ChatGPT Plus、Pro和Team用户已经可以使用o3、o4-mini和o4-mini-high&#xff0c;取代o1、o3-mini和o3-mini-high。具体特点&#xff1a; ChatGPT-o3 特点&#xff1a;o3模型使用高级推理技术&#xff0c;这意味着它在处理复杂问题和逻辑推理方面表现出色。但是不能联网搜索 …

ESP-ADF外设子系统深度解析:esp_peripherals组件架构与核心设计(输入类外设之触摸屏 Touch)

目录 ESP-ADF外设子系统深度解析&#xff1a;esp_peripherals组件架构与核心设计&#xff08;输入类外设之触摸屏 Touch&#xff09;简介模块概述功能定义架构位置核心特性 触摸(Touch)外设触摸外设概述触摸外设API和数据结构外设层API&#xff08;periph_touch.h/periph_touch…

python 读取分级目录

import osdef read_files_in_directory(root_dir):# 遍历根目录下的所有文件和目录for year_dir in os.listdir(root_dir):year_path os.path.join(root_dir, year_dir)if os.path.isdir(year_path): # 确保是目录for month_dir in os.listdir(year_path):# if month_dir in …

MongoServerError: Authentication failed.处理办法

1停止MongoDB服务&#xff1a; systemctl stop mongod2临时修改MongoDB配置&#xff0c;禁用认证&#xff1a; vim /etc/mongdb.config 在配置文件中找到 security:authorization: disabled # 临时关闭认证3.重启MongoDB服务 # 重启MongoDB服务 sudo systemctl restart mon…

ObjectInputStream 终极解析与记忆指南

ObjectInputStream 终极解析与记忆指南 一、核心本质 ObjectInputStream 是 Java 提供的对象反序列化流,继承自 InputStream,用于读取由ObjectOutputStream序列化的Java对象。 核心特性速查表 特性说明继承链InputStream → ObjectInputStream核心功能实现Java对象反序列化…

Java面试高频问题(1-5)

一、HashMap实现原理与并发问题 核心机制 1. 哈希冲突解决方案&#xff1a;采用数组链表红黑树结构&#xff08;JDK1.8&#xff09;&#xff0c;当链表长度超过阈值&#xff08;默认8&#xff09;时转为红黑树&#xff0c;提升查询效率 2. 扩容机制&#xff1a;当元素数量超过…

Genspark:重新定义AI搜索与代理的全能型工具

在当今快速发展的AI技术领域&#xff0c;搜索工具正在经历前所未有的变革。Genspark&#xff0c;这家由前百度高管景鲲和朱凯华创立的AI公司&#xff0c;为我们带来了全新的AI代理引擎体验。作为一位专注于AI工具分享的博主&#xff0c;今天我将为大家详细介绍这款强大的工具&a…

工作记录3

前言: 继续刷尚硅谷的前端视频,查漏补缺。 JS (1)apply() 方法与 call() 方法 (2)构造函数 (3)原型对象<