Linux下搭建HTTP服務(wù)器完成圖片顯示功能
1. 前言
作為Linux下socket(TCP)網(wǎng)絡(luò)編程的練習(xí),使用C語言代碼搭建一個簡單的HTTP服務(wù)器,完成與瀏覽器之間的交互,最終在瀏覽器上顯示一張圖片。
2. HTTP協(xié)議介紹
HTTP協(xié)議本身是基于TCP通信協(xié)議來傳遞數(shù)據(jù)(HTML 文件, 圖片文件-也叫超文本傳輸協(xié)議),HTTP協(xié)議必須工作在客戶端-服務(wù)端架構(gòu)上(本身底層就是TCP),HTTP 默認(rèn)端口號為 80(瀏覽器訪問默認(rèn)就是80端口),但是你也可以改為 8080 或者其他端口(可以手動指定端口)。
HTTP協(xié)議是無連接的,也就是限制每次連接只處理一個請求;服務(wù)器處理完客戶的請求,并收到客戶的應(yīng)答后,即斷開連接。采用這種方式可以節(jié)省傳輸時間。
3. HTTP的消息結(jié)構(gòu)
客戶端向HTTP服務(wù)器發(fā)送的請求消息格式包括了4個部分:請求行(request line)、 請求頭部(header)、空行、請求數(shù)據(jù)
下面這個是瀏覽器的請求,可以對比上面這張圖的格式:
GET / HTTP/1.1 Host: 10.0.0.6 Connection: keep-alive Upgrade-Insecure-Requests: 1 User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9 Accept-Encoding: gzip, deflate Accept-Language: zh-CN,zh;q=0.9
HTTP常用的請求是GET和POST 。
HTTP1.0 定義了三種請求方法: GET, POST 和 HEAD 方法。
HTTP1.1 新增了五種請求方法: OPTIONS, PUT, DELETE, TRACE 和 CONNECT 方法。
HTTP服務(wù)器向客戶端的響應(yīng)也由四個部分組成,分別是:狀態(tài)行、消息報頭、空行、響應(yīng)正文。
例如:
"HTTP/1.1 200 OK\r\n" "Content-type:image/jpeg\r\n" "Content-Length:1234\r\n" "\r\n" "...............正文............."
上面列出的報文字段含義:
HTTP/1.0 200 OK: Http/1.0 表示當(dāng)前協(xié)議為 Http。 1.0 是協(xié)議的版本。 200 表示成功
Content-type : 告訴瀏覽器回送的數(shù)據(jù)類型
Content-Length: 告訴瀏覽器報文中實體主體的大小,也就是返回的內(nèi)容長度
上面字段里回復(fù)的狀態(tài)碼一般有好幾種,分別是:
200 - 請求成功
301 - 資源(網(wǎng)頁等)被永久轉(zhuǎn)移到其它 URL
404 - 請求的資源(網(wǎng)頁等)不存在
500 - 內(nèi)部服務(wù)器錯誤
4. HTTP交互流程
第一次請求是由HTTP客戶端(瀏覽器)發(fā)起的,HTTP服務(wù)器收到請求后,對請求進(jìn)行解析,然后完成后續(xù)的交互。
如果要在瀏覽器上顯示一張圖片,那么交互的流程大致如下:
要讓瀏覽器在界面顯示一張圖片,還得編寫一個HTML代碼給瀏覽器,直接用一個圖片標(biāo)簽即可。
當(dāng)前程序使用的HTML代碼比較簡單,代碼下面貼出來了:
<! DOCTYPE HTML> <html> <head> <title>jpg</title> <meta http-equiv="content-type" content="text/html; charset=iso-8859-1" /> </head> <body> <center><img src="www/123.jpg" width="512px" height="384px" /> </center> </body> </html>
然后還得準(zhǔn)備一張JPG圖片,作為資源文件,方便傳遞給瀏覽器,本地文件結(jié)構(gòu)如下:
5. 案例代碼: 搭建HTTP服務(wù)器
下面代碼采用多線程形式響應(yīng)瀏覽器的請求。
#include <stdio.h> #include <unistd.h> #include <string.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <dirent.h> #include <stdlib.h> #include <pthread.h> #include <semaphore.h> #include <signal.h> #include <sys/types.h> #include <sys/socket.h> #include <arpa/inet.h> #include <netinet/in.h> #include <pthread.h> /* 函數(shù)功能: 服務(wù)器向客戶端發(fā)送響應(yīng)數(shù)據(jù) */ int HTTP_ServerSendFile(int client_fd,char *buff,char *type,char *file) { /*1. 打開文件*/ int fd=open(file,2); if(fd<0)return -1; /*2. 獲取文件大小*/ struct stat s_buff; fstat(fd,&s_buff); /*3. 構(gòu)建響應(yīng)頭部*/ sprintf(buff,"HTTP/1.1 200 OK\r\n" "Content-type:%s\r\n" "Content-Length:%d\r\n" "\r\n",type,s_buff.st_size); /*4. 發(fā)送響應(yīng)頭*/ if(write(client_fd,buff,strlen(buff))!=strlen(buff))return -2; /*5. 發(fā)送消息正文*/ int cnt; while(1) { cnt=read(fd,buff,1024); if(write(client_fd,buff,cnt)!=cnt)return -3; if(cnt!=1024)break; } return 0; } /*線程工作函數(shù)*/ void *thread_work_func(void *argv) { int client_fd=*(int*)argv; free(argv); unsigned int cnt; unsigned char buff[1024]; //讀取瀏覽器發(fā)送過來的數(shù)據(jù) cnt=read(client_fd,buff,1024); buff[cnt]='\0'; printf("%s\n",buff); if(strstr(buff,"GET / HTTP/1.1")) { HTTP_ServerSendFile(client_fd,buff,"text/html","www/image_text.html"); } else if(strstr(buff,"GET /www/123.jpg HTTP/1.1")) { HTTP_ServerSendFile(client_fd,buff,"image/jpeg","www/888.jpg"); } else if(strstr(buff,"GET /favicon.ico HTTP/1.1")) { HTTP_ServerSendFile(client_fd,buff,"image/x-icon","www/1.ico"); } close(client_fd); //退出線程 pthread_exit(NULL); } int main(int argc,char **argv) { if(argc!=2) { printf("./app <端口號>\n"); return 0; } signal(SIGPIPE,SIG_IGN); //忽略 SIGPIPE 信號--防止服務(wù)器異常退出 int sockfd; /*1. 創(chuàng)建socket套接字*/ sockfd=socket(AF_INET,SOCK_STREAM,0); int on = 1; setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)); /*2. 綁定端口號與IP地址*/ struct sockaddr_in addr; addr.sin_family=AF_INET; addr.sin_port=htons(atoi(argv[1])); // 端口號0~65535 addr.sin_addr.s_addr=INADDR_ANY; //inet_addr("0.0.0.0"); //IP地址 if(bind(sockfd,(const struct sockaddr *)&addr,sizeof(struct sockaddr))!=0) { printf("服務(wù)器:端口號綁定失敗.\n"); } /*3. 設(shè)置監(jiān)聽的數(shù)量,表示服務(wù)器同一時間最大能夠處理的連接數(shù)量*/ listen(sockfd,20); /*4. 等待客戶端連接*/ int *client_fd; struct sockaddr_in client_addr; socklen_t addrlen; pthread_t thread_id; while(1) { addrlen=sizeof(struct sockaddr_in); client_fd=malloc(sizeof(int)); *client_fd=accept(sockfd,(struct sockaddr *)&client_addr,&addrlen); if(*client_fd<0) { printf("客戶端連接失敗.\n"); return 0; } printf("連接的客戶端IP地址:%s\n",inet_ntoa(client_addr.sin_addr)); printf("連接的客戶端端口號:%d\n",ntohs(client_addr.sin_port)); /*創(chuàng)建線程*/ if(pthread_create(&thread_id,NULL,thread_work_func,client_fd)) { printf("線程創(chuàng)建失敗.\n"); break; } /*設(shè)置線程的分離屬性*/ pthread_detach(thread_id); } /*5. 關(guān)閉連接*/ close(sockfd); return 0; }
6. 最終運(yùn)行的效果
到此這篇關(guān)于Linux下搭建簡易的HTTP服務(wù)器完成圖片顯示的文章就介紹到這了,更多相關(guān)linux搭建http服務(wù)器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
服務(wù)器遠(yuǎn)程超出最大連接數(shù)的原因及解決
服務(wù)器遠(yuǎn)程超出最大連接數(shù)的原因是沒有注銷退出有可能導(dǎo)致遠(yuǎn)程連接的進(jìn)程死掉,重啟服務(wù)器即可2014-05-05阿里云k8s服務(wù)springboot項目應(yīng)用升級時出現(xiàn)502錯誤
這篇文章主要介紹了阿里云k8s服務(wù)springboot項目應(yīng)用升級時出現(xiàn)502錯誤,需要的朋友可以參考下2022-04-04xampp中apache在windows7下無法啟動的解決方法
下載zip版本的xampp,無法啟動apache ,現(xiàn)象是短暫顯示了running標(biāo)志之后就停止了,日志顯示W(wǎng)ARNING:terminating worker thread 0。2010-03-03Vestacp整合WHMCS實現(xiàn)自動銷售開通虛擬主機(jī)服務(wù)教程
這篇文章主要為大家詳細(xì)介紹了Vestacp整合WHMCS實現(xiàn)自動銷售開通虛擬主機(jī)服務(wù)教程,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07FileZilla Server搭建FTP服務(wù)器配置及425錯誤與TLS警告解決方法詳解
本文詳細(xì)講解了FileZilla Server搭建FTP服務(wù)器配置以及425 Can't open data,You appear to be behind a NAT router,FTP over TLS is not enabled等相關(guān)問題的解決方法2018-10-10