一次完整的 HTTP 請求過程

原文:http://blog.jobbole.com/106632/

一次完整的HTTP請求過程從TCP三次握手建立連接成功後開始,客戶端按照指定的格式開始向服務端發送HTTP請求,服務端接收請求後,解析HTTP請求,處理完業務邏輯,最後返回一個HTTP的響應給客戶端,HTTP的響應內容同樣有標準的格式。無論是什麼客戶端或者是什麼服務端,大家只要按照HTTP的協議標準來實現的話,那麼它一定是通用的。

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響應消息的格式包括:狀態行、響應頭、空行、消息體。每部分內容佔一行。

<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-format

狀態行:狀態行位於相應消息的第一行,有HTTP協議版本號,狀態碼和狀態說明三部分構成。如:

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服務器了。代碼:

-- 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()

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

-- 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

http

參考:

Response
Request
Scoket

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章