《TCP IP網絡編程》第二十四章

第 24 章 制作 HTTP 服務器端

24.1 HTTP 概要

????????本章將編寫 HTTP(HyperText Transfer Protocol,超文本傳輸協議)服務器端,即 Web 服務器端

理解 Web 服務器端:

????????web服務器端就是要基于 HTTP 協議,將網頁對應文件傳輸給客戶端的服務器端。Hypertext(超文本)是可以根據客戶端請求而跳轉的結構化信息。

HTTP:

? ? ? ? HTTP協議是以超文本傳輸為目的而設計的應用層協議,這種協議同樣屬于基于TCP/IP實現的協議,從結果上看,實現該協議相當于實現web服務器端。

????????此外,瀏覽器也屬于基于套接字的客戶端,因為連接到任意Web服務器端時,瀏覽器內部也會創建套接字,只不過瀏覽器多了一項功能:將服務器端傳輸的HTML格式的超文本解析為可讀性較強的視圖。 總之,Web服務器端是以HTTP協議為基礎傳輸超文本的服務器端

? ? ? ? HTTP是無狀態的 Stateless 協議。如圖:?

????????從上圖可以看出,服務器端相應客戶端請求后立即斷開連接。換言之,服務器端不會維持客戶端狀態。即使同一客戶端再次發送請求,服務器端也無法辨認出是原先那個,而會以相同方式處理新請求。因此,HTTP 又稱「無狀態的 Stateless 協議」。

請求消息(Request Message)的結構:

????????下面是客戶端向服務端發起請求消息的結構:

????????從圖中可以看出,請求消息可以分為請求頭、消息頭、消息體 3 個部分。其中,請求行含有請求方式(請求目的)信息。典型的請求方式有 GET 和 POST ,GET 主要用于請求數據,POST 主要用于傳輸數據。為了降低復雜度,我們實現只能響應 GET 請求的 Web 服務器端,下面解釋圖中的請求行信息。其中「GET/index.html HTTP/1.1」 具有如下含義:?

????????請求(GET)index.html 文件,通常以 1.1 版本的 HTTP 協議進行通信。

????????請求行只能通過 1 行(line)發送,因此,服務器端很容易從 HTTP 請求中提取第一行,并分別分析請求行中的信息。

????????請求行下面的消息頭中包含發送請求的瀏覽器信息、用戶認證信息等關于 HTTP 消息的附加信息。最后的消息體中裝有客戶端向服務端傳輸的數據,為了裝入數據,需要以 POST 方式發送請求。但是我們的目標是實現 GET 方式的服務器端,所以可以忽略這部分內容。另外,消息體和消息頭與之間以空行隔開,因此不會發生邊界問題

響應消息(Response Message)的結構:

????????下面是 Web 服務器端向客戶端傳遞的響應信息的結構。從圖中可以看出,該響應消息由狀態行、頭信息、消息體等 3 個部分組成。狀態行中有關于請求的狀態信息,這是與請求消息相比最為顯著地區別。

????????第一個字符串狀態行中含有關于客戶端請求的處理結果。例如,客戶端請求 index.html 文件時,表示 index.html 文件是否存在、服務端是否發生問題而無法響應等不同情況的信息寫入狀態行。圖中的「HTTP/1.1 200 OK」具有如下含義:

? ? ? ? 我想用HTTP1.1版本進行響應,你的請求已正確處理(200,OK)。

? ? ? ? 表示“客戶端請求的執行結果”的數字成為狀態碼,典型的有以下幾種:

  • 200 OK : 成功處理了請求!
  • 404 Not Found : 請求的文件不存在!
  • 400 Bad Request : 請求方式錯誤,請檢查!

????????消息頭中含有傳輸的數據類型和長度等信息。圖中的消息頭含有如下信息:

???服務端名為 SimpleWebServer ,傳輸的數據類型為 text/html。數據長度不超過 2048 個字節。

????????最后插入一個空行后,通過消息體發送客戶端請求的文件數據。以上就是實現 Web 服務端過程中必要的 HTTP 協議。

24.2 實現簡單的 Web 服務器端

實現基于 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);// 檢查請求行中是否包含 "HTTP/"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));// 檢查請求方法是否為 "GET"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);
}

運行結果:

????????經過測試,這個簡單的 HTTP 服務器可以正常的顯示出頁面。?

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/43545.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/43545.shtml
英文地址,請注明出處:http://en.pswp.cn/news/43545.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

easyx圖形庫基礎:3實現彈球小游戲

實現彈球小游戲 一.實現彈球小游戲:1.初始化布&#xff1a;2.初始化一個球的信息&#xff1a;3.球的移動和碰撞反彈4.底邊擋板的繪制和移動碰撞重置數據。 二.整體代碼&#xff1a; 一.實現彈球小游戲: 1.初始化布&#xff1a; int main() {initgraph(800, 600);setorigin(40…

[論文筆記]Glancing Transformer for Non-Autoregressive Neural Machine Translation

引言 這是論文Glancing Transformer for Non-Autoregressive Neural Machine Translation的筆記。 傳統的非自回歸文本生成速度較慢,因為需要給定之前的token來預測下一個token。但自回歸模型雖然效率高,但性能沒那么好。 這篇論文提出了Glancing Transformer,可以只需要一…

layui下拉框select 彈出層在最外層

出現問題如圖所示 想要的效果是如下 這樣的效果只需一行代碼就能解決 .layui-layer-page .layui-layer-content{overflow: visible!important;}

Postgresql源碼(112)plpgsql執行sql時變量何時替換為值

相關 《Postgresql源碼&#xff08;41&#xff09;plpgsql函數編譯執行流程分析》 《Postgresql源碼&#xff08;46&#xff09;plpgsql中的變量類型及對應關系》 《Postgresql源碼&#xff08;49&#xff09;plpgsql函數編譯執行流程分析總結》 《Postgresql源碼&#xff08;5…

PyTorch從零開始實現ResNet

文章目錄 代碼實現參考 代碼實現 本文實現 ResNet原論文 Deep Residual Learning for Image Recognition 中的50層&#xff0c;101層和152層殘差連接。 代碼中使用基礎殘差塊這個概念&#xff0c;這里的基礎殘差塊指的是上圖中紅色矩形圈出的內容&#xff1a;從上到下分別使用…

感覺和身邊其他人有差距怎么辦?

雖然清楚知識需要靠時間沉淀&#xff0c;但在看到自己做不出來的題別人會做&#xff0c;自己寫不出的代碼別人會寫時還是會感到焦慮怎么辦&#xff1f; 你是否也因為自身跟周圍人的差距而產生過迷茫&#xff0c;這份迷茫如今是被你克服了還是仍舊讓你感到困擾&#xff1f; 下…

LabVIEW開發最小化5G系統測試平臺

LabVIEW開發最小化5G系統測試平臺 由于具有大量存儲能力和數據的應用程序的智能手機的激增&#xff0c;當前一代產品被迫提高其吞吐效率。正交頻分復用由于其卓越的品質&#xff0c;如單抽頭均衡和具有成本效益的實施&#xff0c;現在被廣泛用作物理層技術。這些好處是以嚴格的…

ElasticSearch索引庫、文檔、RestClient操作

文章目錄 一、索引庫1、mapping屬性2、索引庫的crud 二、文檔的crud三、RestClient 一、索引庫 es中的索引是指相同類型的文檔集合&#xff0c;即mysql中表的概念 映射&#xff1a;索引中文檔字段的約束&#xff0c;比如名稱、類型 1、mapping屬性 mapping映射是對索引庫中文…

Elasticsearch在部署時,對Linux的設置有哪些優化方法?

部署Elasticsearch時&#xff0c;可以通過優化Linux系統的設置來提升性能和穩定性。以下是一些常見的優化方法&#xff1a; 1.文件描述符限制 Elasticsearch需要大量的文件描述符來處理數據和連接&#xff0c;所以確保調整系統的文件描述符限制。可以通過修改 /etc/security/…

Docker-compose搭建Git私服

1. 新建個專用的目錄&#xff0c;然后在里面新建個docker-compose.yml文件&#xff1a; &#xff08;gitlab-ce是社區版&#xff0c;當然還有ee&#xff0c;是商業版&#xff09; version: 3.6 …

es自定義分詞器支持數字字母分詞,中文分詞器jieba支持添加禁用詞和擴展詞典

自定義分析器&#xff0c;分詞器 PUT http://xxx.xxx.xxx.xxx:9200/test_index/ {"settings": {"analysis": {"analyzer": {"char_test_analyzer": {"tokenizer": "char_test_tokenizer","filter": [&…

公網遠程連接Redis數據庫詳解

文章目錄 1. Linux(centos8)安裝redis數據庫2. 配置redis數據庫3. 內網穿透3.1 安裝cpolar內網穿透3.2 創建隧道映射本地端口 4. 配置固定TCP端口地址4.1 保留一個固定tcp地址4.2 配置固定TCP地址4.3 使用固定的tcp地址連接 前言 潔潔的個人主頁 我就問你有沒有發揮&#xff0…

ssh免密登陸報錯ERROR: @ WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!

問題描述&#xff1a; 在日常的運維中需要做ssh的免密登陸有提示如下的報錯內容&#xff1a; [rootpaas-harbor01 cce-v5.2.3]# ssh-copy-id 192.45.66.14 /usr/bin/ssh-copy-id: INFO: Source of key(s) to be installed: "/root/.ssh/id_rsa.pub" /usr/bin/ssh-c…

通訊錄實現【C語言】

目錄 前言 一、整體邏輯分析 二、實現步驟 1、創建菜單和多次操作問題 2、創建通訊錄 3、初始化通訊錄 4、添加聯系人 5、顯示聯系人 6、刪除指定聯系人 ?7、查找指定聯系人 8、修改聯系人信息 9、排序聯系人信息 三、全部源碼 前言 我們上期已經詳細的介紹了自定…

Java SpringBoot Vue ERP系統

系統介紹 該ERP系統基于SpringBoot框架和SaaS模式&#xff0c;支持多租戶&#xff0c;專注進銷存財務生產功能。主要模塊有零售管理、采購管理、銷售管理、倉庫管理、財務管理、報表查詢、系統管理等。支持預付款、收入支出、倉庫調撥、組裝拆卸、訂單等特色功能。擁有商品庫存…

ubuntu設置共享文件夾成功后卻不顯示找不到(已解決)

1.首先輸下面命令查看是否真的設置成功共享文件夾 vmware-hgfsclient如果確實已經設置過共享文件夾將輸出window下共享文件夾名字 2.確認自己已設置共享文件夾后輸入下面的命令 //如果之前沒有命令包則先執行sudo apt-get install open-vm-tools sudo vmhgfs-fuse .host:/ /mn…

十六、Spring Cloud Sleuth 分布式請求鏈路追蹤

目錄 一、概述1、為什么出出現這個技術&#xff1f;需要解決哪些問題2、是什么&#xff1f;3、解決 二、搭建鏈路監控步驟1、下載運行zipkin2、服務提供者3、服務調用者4、測試 一、概述 1、為什么出出現這個技術&#xff1f;需要解決哪些問題 2、是什么&#xff1f; 官網&am…

spss---如何使用信度分析以及案例分析

信度分析 問卷調查法是教育研究中廣泛采用的一種調查方法&#xff0c;根據調查目的設計的調查問卷是問卷調查法獲取信息的工具&#xff0c;其質量高低對調查結果的真實性、適用性等具有決定性的作用。 為了保證問卷具有較高的可靠性和有效性&#xff0c;在形成正式問卷之 前&…

CLion:最好用的c/c++編寫工具(最詳細安裝教程)

目錄 一.前言介紹 1.下載安裝 1.1右上角點擊下載 1.2選擇自己操作系統&#xff0c;然后點擊下載 1.3選擇next 1.4 更改路徑 1.5D盤最好 1.6 按照我的選擇配置環境 1.7install安裝 1.8 安裝完成 2、mingw64安裝 2.1下載資源壓縮包 2.2mingw64放入到合適的位置&#xff0c;…

Redis五大基本數據類型及其使用場景

文章目錄 **一 什么是NoSQL&#xff1f;****二 redis是什么&#xff1f;****三 redis五大基本類型**1 String&#xff08;字符串&#xff09;**應用場景** 2 List&#xff08;列表&#xff09;**應用場景** 3 Set&#xff08;集合&#xff09;4 sorted set&#xff08;有序集合…