python實現(xiàn)靜態(tài)web服務器
HTTP協(xié)議簡介
HTTP請求
1:瀏覽器首先向服務器發(fā)送HTTP請求,請求包括:
方法:GET還是POST,GET僅請求資源,POST會附帶用戶數(shù)據(jù);
路徑:/full/url/path;
域名:由Host頭指定:Host: www.sina.com以及其他相關(guān)的Header;
如果是POST,那么請求還包括一個Body,包含用戶數(shù)據(jù)
2:服務器向瀏覽器返回HTTP響應,響應包括:
響應代碼:200表示成功,3xx表示重定向,4xx表示客戶端發(fā)送的請求有錯誤,5xx表示服務器端處理時發(fā)生了錯誤;
響應類型:由Content-Type指定;
以及其他相關(guān)的Header;
通常服務器的HTTP響應會攜帶內(nèi)容,也就是有一個Body,包含響應的內(nèi)容,網(wǎng)頁的HTML源碼就在Body中。
3:如果瀏覽器還需要繼續(xù)向服務器請求其他資源,比如圖片,就再次發(fā)出HTTP請求,重復步驟1、2。
Web采用的HTTP協(xié)議采用了非常簡單的請求-響應模式,從而大大簡化了開發(fā)。當我們編寫一個頁面時,我們只需要在HTTP請求中把HTML發(fā)送出去,不需要考慮如何附帶圖片、視頻等,瀏覽器如果需要請求圖片和視頻,它會發(fā)送另一個HTTP請求,因此,一個HTTP請求只處理一個資源(此時就可以理解為TCP協(xié)議中的短連接,每個鏈接只獲取一個資源,如需要多個就需要建立多個鏈接)
HTTP格式
每個HTTP請求和響應都遵循相同的格式,一個HTTP包含Header和Body兩部分,其中Body是可選的。
HTTP協(xié)議是一種文本協(xié)議,所以,它的格式也非常簡單。
1 HTTP GET請求的格式:
GET /path HTTP/1.1 Header1: Value1 Header2: Value2 Header3: Value3
每個Header一行一個,換行符是\r\n。
2 HTTP POST請求的格式:
POST /path HTTP/1.1 Header1: Value1 Header2: Value2 Header3: Value3 body data goes here...
3 HTTP響應的格式:
200 OK Header1: Value1 Header2: Value2 Header3: Value3 body data goes here...
HTTP響應如果包含body,也是通過\r\n\r\n來分隔的。
請再次注意,Body的數(shù)據(jù)類型由Content-Type頭來確定,如果是網(wǎng)頁,Body就是文本,如果是圖片,Body就是圖片的二進制數(shù)據(jù)。
當存在Content-Encoding時,Body數(shù)據(jù)是被壓縮的,最常見的壓縮方式是gzip,所以,看到Content-Encoding: gzip時,需要將Body數(shù)據(jù)先解壓縮,才能得到真正的數(shù)據(jù)。壓縮的目的在于減少Body的大小,加快網(wǎng)絡(luò)傳輸。
demo:靜態(tài)web服務器
import socket
import re
import time
def service_client(new_socket):
"""為這個客戶端服務"""
# 1.接收瀏覽器發(fā)送過來的請求,即http請求
# GET / HTTP/1.1
# --------
request = new_socket.recv(1024).decode('utf-8')
# 判斷客戶端意外斷開鏈接返回空字符串
if not request:
# 關(guān)閉套接字并退出
new_socket.close()
print("==="*30)
return
# 分隔套接字
request_lines = request.splitlines()
print()
print(">"*20)
print(request_lines)
file_name = ""
ret = re.match(r"[^/]+(/[^ ]*)", request_lines[0])
if ret:
file_name = ret.group(1)
if file_name == "/":
file_name = "/index.html"
# 2.返回http格式數(shù)據(jù) 給瀏覽器
try:
f = open("./html" + file_name, "rb")
except:
response = "HTTP/1.1 404 NOT FOUND\r\n"
response += "Content-Type:text/html;charset=utf-8\r\n"
response += "\r\n"
response += "<h1>404 not found <br> 沒有發(fā)現(xiàn)所請求資源</h1>"
response += str(time.strftime('%Y-%m-%d %H:%M:%S', time.localtime(time.time())))
new_socket.send(response.encode('utf-8'))
else:
html_content = f.read()
f.close()
# 2.1準備發(fā)送給瀏覽器的數(shù)據(jù)---header
response = "HTTP/1.1 200 OK\r\n"
response += "\r\n"
# 2.2準備發(fā)送給瀏覽器的數(shù)據(jù) ---body
# 將response header發(fā)送給瀏覽器
new_socket.send(response.encode("utf-8"))
# 將response body發(fā)送給瀏覽器
new_socket.send(html_content)
# 關(guān)閉套接字
new_socket.close()
def main():
"""用來完成整體的控制"""
# 1.創(chuàng)建套接字
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 端口復用
tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
# 2.綁定
tcp_server_socket.bind(("", 7890))
# 3.變?yōu)樘捉幼?
tcp_server_socket.listen(128)
while True:
# 4.等待客戶端的鏈接
new_socket, client_addr = tcp_server_socket.accept()
print(client_addr)
# 5.為這個客戶端服務
service_client(new_socket)
# 關(guān)閉監(jiān)聽套接字
tcp_server_socket.close()
if __name__ == '__main__':
main()
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
virtualenv隔離Python環(huán)境的問題解析
virtualenv為應用提供了隔離的Python運行環(huán)境,解決了不同應用間多版本的沖突問題,這篇文章主要介紹了virtualenv隔離Python環(huán)境,需要的朋友可以參考下2022-06-06
Pandas數(shù)據(jù)類型轉(zhuǎn)換df.astype()及數(shù)據(jù)類型查看df.dtypes的使用
Python,numpy都有自己的一套數(shù)據(jù)格式,本文主要介紹了Pandas數(shù)據(jù)類型轉(zhuǎn)換df.astype()及數(shù)據(jù)類型查看df.dtypes的使用,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-07-07
Python+Selenium實現(xiàn)表單自動填充和提交
你是不是也厭倦了每天重復表單填寫的工作,是時候讓技術(shù)來幫助我們解放雙手了,下面小編就為大家介紹一下如何使用Selenium和Python來自動填充和提交表單2023-09-09

