原文:http://blog.jobbole.com/106632/html
一次完整的HTTP請求過程從TCP三次握手創建鏈接成功後開始,客戶端按照指定的格式開始向服務端發送HTTP請求,服務端接收請求後,解析HTTP請求,處理完業務邏輯,最後返回一個HTTP的響應給客戶端,HTTP的響應內容一樣有標準的格式。不管是什麼客戶端或者是什麼服務端,你們只要按照HTTP的協議標準來實現的話,那麼它必定是通用的。nginx
HTTP 請求格式
HTTP請求格式主要有四部分組成,分別是:請求行、請求頭、空行、消息體,每部份內容佔一行
http://jbcdn2.b0.upaiyun.com/2016/10/c0cdafd8bdb8d0c87b3c35498aa0417f.png編程
<request-line>
<general-headers>
<request-headers>
<entity-headers>
<empty-line>
[<message-body>]
1
2
3
4
5
6
<request-line>
<general-headers>
<request-headers>
<entity-headers>
<empty-line>
[<message-body>]
request-format瀏覽器
請求行:請求行是請求消息的第一行,由三部分組成:分別是請求方法(GET/POST/DELETE/PUT/HEAD)、請求資源的URI路徑、HTTP的版本號緩存
GET /index.html HTTP/1.1
1
GET /index.html HTTP/1.1
請求頭:請求頭中的信息有和緩存相關的頭(Cache-Control,If-Modified-Since)、客戶端身份信息(User-Agent)等等。例如:服務器
Cache-Control:max-age=0
Cookie:gsScrollPos=; _ga=GA1.2.329038035.1465891024; _gat=1
If-Modified-Since:Sun, 01 May 2016 11:19:03 GMT
User-Agent:Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/51.0.2704.84 Safari/537.36
1
2
3
4
Cache-Control:max-age=0
Cookie:gsScrollPos=; _ga=GA1.2.329038035.1465891024; _gat=1
If-Modified-Since:Sun, 01 May 2016 11:19:03 GMT
User-Agent:Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/51.0.2704.84 Safari/537.36
消息體:請求體是客戶端發給服務端的請求數據,這部分數據並非每一個請求必須的。網絡
HTTP 響應格式
服務器接收處理完請求後返回一個HTTP相應消息給客戶端。HTTP響應消息的格式包括:狀態行、響應頭、空行、消息體。每部份內容佔一行。socket
<status-line>
<general-headers>
<response-headers>
<entity-headers>
<empty-line>
[<message-body>]
1
2
3
4
5
6
<status-line>
<general-headers>
<response-headers>
<entity-headers>
<empty-line>
[<message-body>]
response-formatide
狀態行:狀態行位於相應消息的第一行,有HTTP協議版本號,狀態碼和狀態說明三部分構成。如:code
HTTP/1.1 200 OK
1
HTTP/1.1 200 OK
響應頭:響應頭是服務器傳遞給客戶端用於說明服務器的一些信息,以及未來繼續訪問該資源時的策略。
Connection:keep-alive
Content-Encoding:gzip
Content-Type:text/html; charset=utf-8
Date:Fri, 24 Jun 2016 06:23:31 GMT
Server:nginx/1.9.12
Transfer-Encoding:chunked
1
2
3
4
5
6
Connection:keep-alive
Content-Encoding:gzip
Content-Type:text/html; charset=utf-8
Date:Fri, 24 Jun 2016 06:23:31 GMT
Server:nginx/1.9.12
Transfer-Encoding:chunked
響應體:響應體是服務端返回給客戶端的HTML文本內容,或者其餘格式的數據,好比:視頻流、圖片或者音頻數據。
Socket
WEB Server都是基於Socket編程,又稱之爲網絡編程,網絡協議經過一個叫作socket的對象抽象出來,socket能夠創建網絡鏈接,讀數據,寫數據。socket模塊定義了一些常量參數,用來指定socket的的地址族、socket的類型、以及支持的TCP/IP協議。
socket.socket([family[, type[, proto]]]):根據指定的地址族和套接字類型、協議編號(默認爲0)來建立套接字對象。AF_INET對應的IPV4, AF_INET6對應的IPV6。
%e8%a1%a81
Socket 對象方法
socket.bind(address):綁定IP地址以及端口
socket.listen(backlog) :在指定的端口開始監聽,backlog表示connection隊列的最大長度
socket.setblocking(flag) : 設置爲非阻塞仍是阻塞的socket,若是是非阻塞的,那麼調用recv的時候若是沒有數據可讀,那麼久直接返回一個錯誤,相反若是設置爲阻塞模式,若是沒有數據可讀,那麼就一直處於阻塞等待數據的狀態。
socket.accept():當有鏈接請求過來時,接收該鏈接,返回一個socket對象,該對象能夠在基於該鏈接發送和接收數據。
socket.sendall(string[, flags]):發送數據
socket.recv(bufsize[, flags]):接收數據
socket.close():關閉socket鏈接。
搞清楚了HTTP規範和Socket以後,咱們就可使用Socket實現一個對簡單的HTTP服務器了。代碼:
import socket
if name == 'main':
PORT = 8000
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.bind(('127.0.0.1', PORT))
sock.listen(1)
print 'Serving HTTP on port %s ...' % PORT
while 1: conn, addr = sock.accept() print conn, addr request = conn.recv(1024) # HTTP響應消息 response = "HTTP/1.1 200 OK\nContent-Type:text/html\nServer:myserver\n\nHello, World!" conn.sendall(response) conn.close()
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import socket
if name == 'main':
PORT = 8000
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.bind(('127.0.0.1', PORT))
sock.listen(1)
print 'Serving HTTP on port %s ...' % PORT
while 1: conn, addr = sock.accept() print conn, addr request = conn.recv(1024) # HTTP響應消息 response = "HTTP/1.1 200 OK\nContent-Type:text/html\nServer:myserver\n\nHello, World!" conn.sendall(response) conn.close()
瀏覽器訪問地址:http://localhost:8000
http
參考:
ResponseRequestScoket