一、引言
歡迎大家接着看我的博客,如何大家有什么想法的話回復我哦,閑話不多聊了,接着上一講的內容來說吧,在上一節中已經講到了請求頭字符串的解析,並且在解析中我我們已經獲取了url。就是上節中提到的/doing。當瀏覽器發送了/doing請求后,這是的與服務器的連接並沒有中斷,原因無他,就是瀏覽器等待接受服務端發來的信息,然后他對信息進行解析並顯示在瀏覽器界面上。在編寫Http服務器時的C/S結構時,我只編寫服務端server,客戶端直接用現成的瀏覽器即可。
當服務端找到瀏覽器需要的內容的時候,現在服務端就需要把內容發送給瀏覽器即可。但是在HTTP協議中,發送具體內容之前,服務端還是要發送一個響應頭,之前瀏覽器發送的叫做請求頭,那么服務端也要發送一個響應頭告訴瀏覽器,說你的請求我接受或者拒接。拒絕的話之后就沒有具體實體內容發送,否則再告訴瀏覽器說你請求的內容的格式是html或者text,或者是音頻等二進制文件等,並且告訴瀏覽器你請求內容的大小(這個非常重要)。當響應頭發送完畢后,如果是同意瀏覽器的請求的話,那么接着發送具體的內容即可,接着關閉這個鏈接。
二、HTTP協議
現在我們來看一下這個響應頭的一個例子,還是只說明一些我們現在需要知道的,其他內容讀者可以自己去查找相關的內容。
1 HTTP/1.1 200 OK 2 Server:(Unix) 3 Content-type:application/octet-stream 4 Content-length:65144369 5 Connection:Keep-Alive
在這個例子中的響應頭我們可以看到:
1)在第一行就告訴瀏覽器http協議的版本是1.1,狀態碼是200,意思是OK,就是說你的請求我同意了。
2)在第二行就是告知瀏覽器我服務器的名字。
3)在第三行就是你請求的內容的格式是applicaiton/octet-stream,理解為二進制流即可。
4)在第四行說明你請求內容的大小,單位是字節。
5)第五行告訴瀏覽器是Keep-Alive連接方式。
在這個協議頭發送給瀏覽器時,還是每行結尾都有一個回車換行\r\n。最后還是需要一個空行代表本響應頭結束。還有一些協議內容到用用到時再講,現在提出這幾項即可。當這個協議頭發送給瀏覽器后,瀏覽器對其解析。然后等待真正的請求內容。當服務端把瀏覽器的請求的具體內容發送完畢就關閉這個連接。
講到這里基本一個請求響應結束了。那在這里提一下如何處理大並發呢。其實在HTTP請求中,在每個URL請求中服務器並不關心一些請求是否來着同個瀏覽器,服務器做的只是有請求來臨就對其連接發送數據,然后關閉數據。盡管可能來着同一客戶端上的請求有數據上的關系,但是服務端只是在發送數據而已。所以對於同一個瀏覽器發送的多個請求和不同瀏覽器或者不同主機發送來的請求的對待都是一樣的。就是每個請求都是獨立的,並不關心這個邏輯關系。
在這一篇中的主要內容也就講完了,好像看起來比較短哦,哈哈。。。。。那么在講一些細節上的內容吧。
現在大並發的服務器linux中,主要在有線程,進程,select,poll,epoll的方式。這些方式在各個博客上都是有的,所以我並不想在解釋一下,讀者可以自己查詢其中的不同之處。而本文開發的HTTP服務器采用線程的方式進行。主要是當一個請求來到時,就創建一個線程來處理。采用的linux下POSIX標准pthtread線程。如果其他讀者可以采用其他的幾種方式來處理。或者自己寫個線程池來取代每次一個請求來創建一個線程。主要對其包裹在一個命名空間為MultiThread的線程Thread類中,接着是一些代碼段:
頭文件(include/thread.h):
1 /* 2 * include/thread.h 3 */ 4 5 #ifndef THREAD_H_ 6 #define THREAD_H_ 7 #include<pthread.h> 8 #include<iostream> 9 namespace MultiThread { 10 11 class Thread { 12 public: 13 Thread(); 14 ~Thread(); 15 static int create_thread(pthread_t &pid,void* (*pFunc)(void *),void *arg); 16 int stop_thread(); 17 18 }; 19 20 } /* namespace MultiThread */ 21 22 #endif /* THREAD_H_ */
cpp文件(src/thread.cpp)
1 /** 2 * 3 * src/thread.cpp 4 **/ 5 #include "thread.h" 6 namespace MultiThread { 7 8 Thread::Thread() { 9 } 10 11 Thread::~Thread() { 12 } 13 14 int Thread::create_thread(pthread_t &pid,void* (*pFunc)(void *),void *arg) { 15 int ret=pthread_create(&pid,NULL,pFunc,arg); 16 if(ret==0) 17 std::cout<<"create_thread() ...succeed"<<std::endl; 18 else 19 std::cout<<"create_thread() ...failed"<<std::endl; 20 return ret; 21 } 22 23 int Thread::stop_thread() { 24 return 0; 25 } 26 27 } /* namespace MultiThread */
看到現在,大家一定發現stop_thread()為空,應為我到現在還沒用這個函數。並且到現在只用到create_thread函數,之后如果用到,接着在之后的文章中提出吧。
如果大家對HTTP服務器感興趣的話,接着看《Http服務器實現文件上傳與下載(三)》和我學習HTTP服務器編寫吧。