一次完整的 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
參考:
相關文章
- 一次完整的HTTP請求過程HTTP
- 完整的一次 HTTP 請求響應過程(二)HTTP
- 完整的一次 HTTP 請求響應過程(一)HTTP
- 一次完整的HTTP請求HTTP
- HTTP的請求過程HTTP
- 【PHP】一次請求過程的解析PHP
- 前端效能優化之http請求的過程前端優化HTTP
- nginx 處理客戶端請求的完整過程Nginx客戶端
- 【PHP-FPM】一次請求過程的解析PHP
- 【進階篇】一文搞清楚網頁發起 HTTP 請求呼叫的完整過程網頁HTTP
- 淺析一次HTTP請求HTTP
- 關於兩次http請求,後一次請求影響前一次請求的問題HTTP
- Web請求過程Web
- YTKNetwork 原始碼閱讀(二)-一次POST請求的完整歷程原始碼
- 一次完整的 Web 請求和渲染過程以及如何優化網頁Web優化網頁
- 走進JavaWeb技術世界5:初探Tomcat的HTTP請求過程JavaWebTomcatHTTP
- http請求到達後端的執行過程——閘道器篇HTTP後端
- 一次HTTP通訊過程HTTP
- http請求HTTP
- HTTP 請求HTTP
- 記一次完整的wordpress安裝過程
- 層層剖析一次 HTTP POST 請求事故HTTP
- 掌握 HTTP 快取——從請求到響應過程的一切(下)HTTP快取
- 掌握 HTTP 快取——從請求到響應過程的一切(上)HTTP快取
- SQL Server儲存過程模擬HTTP請求POST和GET協議SQLServer儲存過程HTTP協議
- Angular 記錄 - Rxjs 完整處理一個 Http 請求AngularJSHTTP
- HBase一次慢查詢請求的問題排查與解決過程
- http請求概述HTTP
- HTTP請求方法HTTP
- http請求頭HTTP
- go http請求GoHTTP
- Laravel 列印請求過程中的所有 SQLLaravelSQL
- django從請求到響應的過程Django
- golang 的 http 請求池GolangHTTP
- 我知道的HTTP請求HTTP
- 合併HTTP請求vs並行HTTP請求,到底誰更快?HTTP並行
- 合併HTTP請求 vs 並行HTTP請求,到底誰更快?HTTP並行
- fastHttp服務端處理請求的過程ASTHTTP服務端