一次完整的 HTTP 請求過程
一次完整的HTTP請求過程從TCP三次握手建立連接成功後開始,客戶端按照指定的格式開始向服務端發送HTTP請求,服務端接收請求後,解析HTTP請求,處理完業務邏輯,最後返回一個HTTP的響應給客戶端,HTTP的響應內容同樣有標準的格式。無論是什麽客戶端或者是什麽服務端,大家只要按照HTTP的協議標準來實現的話,那麽它一定是通用的。
HTTP 請求格式
HTTP請求格式主要有四部分組成,分別是:請求行、請求頭、空行、消息體,每部分內容占一行
<request-line>
<general-headers>
<request-headers>
<entity-headers>
<empty-line>
[<message-body>]
請求行:請求行是請求消息的第一行,由三部分組成:分別是請求方法(GET/POST/DELETE/PUT/HEAD)、請求資源的URI路徑、HTTP的版本號
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
消息體:請求體是客戶端發給服務端的請求數據,這部分數據並不是每個請求必須的。
HTTP 響應格式
服務器接收處理完請求後返回一個HTTP相應消息給客戶端。HTTP響應消息的格式包括:狀態行、響應頭、空行、消息體。每部分內容占一行。
<status-line>
<general-headers>
<response-headers>
<entity-headers>
<empty-line>
[<message-body>]
狀態行:狀態行位於相應消息的第一行,有HTTP協議版本號,狀態碼和狀態說明三部分構成。如:
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
響應體:響應體是服務端返回給客戶端的HTML文本內容,或者其他格式的數據,比如:視頻流、圖片或者音頻數據。
Socket
WEB Server都是基於Socket編程,又稱之為網絡編程,網絡協議通過一個叫做socket的對象抽象出來,socket可以建立網絡連接,讀數據,寫數據。socket模塊定義了一些常量參數,用來指定socket的的地址族、socket的類型、以及支持的TCP/IP協議。
socket.socket([family[, type[, proto]]]):根據指定的地址族和套接字類型、協議編號(默認為0)來創建套接字對象。AF_INET對應的IPV4, AF_INET6對應的IPV6。
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服務器了。代碼:
# -*- coding:utf-8 -*-
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
參考:
- Response
- Request
- Scoket
一次完整的 HTTP 請求過程