HTTP協(xié)議簡(jiǎn)介
HTTP請(qǐng)求
1:瀏覽器首先向服務(wù)器發(fā)送HTTP請(qǐng)求,請(qǐng)求包括:
方法:GET還是POST,GET僅請(qǐng)求資源,POST會(huì)附帶用戶數(shù)據(jù);
路徑:/full/url/path;
域名:由Host頭指定:Host: www.sina.com以及其他相關(guān)的Header;
如果是POST,那么請(qǐng)求還包括一個(gè)Body,包含用戶數(shù)據(jù)
2:服務(wù)器向?yàn)g覽器返回HTTP響應(yīng),響應(yīng)包括:
響應(yīng)代碼:200表示成功,3xx表示重定向,4xx表示客戶端發(fā)送的請(qǐng)求有錯(cuò)誤,5xx表示服務(wù)器端處理時(shí)發(fā)生了錯(cuò)誤;
響應(yīng)類型:由Content-Type指定;
以及其他相關(guān)的Header;
通常服務(wù)器的HTTP響應(yīng)會(huì)攜帶內(nèi)容,也就是有一個(gè)Body,包含響應(yīng)的內(nèi)容,網(wǎng)頁(yè)的HTML源碼就在Body中。
3:如果瀏覽器還需要繼續(xù)向服務(wù)器請(qǐng)求其他資源,比如圖片,就再次發(fā)出HTTP請(qǐng)求,重復(fù)步驟1、2。
Web采用的HTTP協(xié)議采用了非常簡(jiǎn)單的請(qǐng)求-響應(yīng)模式,從而大大簡(jiǎn)化了開(kāi)發(fā)。當(dāng)我們編寫一個(gè)頁(yè)面時(shí),我們只需要在HTTP請(qǐng)求中把HTML發(fā)送出去,不需要考慮如何附帶圖片、視頻等,瀏覽器如果需要請(qǐng)求圖片和視頻,它會(huì)發(fā)送另一個(gè)HTTP請(qǐng)求,因此,一個(gè)HTTP請(qǐng)求只處理一個(gè)資源(此時(shí)就可以理解為TCP協(xié)議中的短連接,每個(gè)鏈接只獲取一個(gè)資源,如需要多個(gè)就需要建立多個(gè)鏈接)
HTTP格式
每個(gè)HTTP請(qǐng)求和響應(yīng)都遵循相同的格式,一個(gè)HTTP包含Header和Body兩部分,其中Body是可選的。
HTTP協(xié)議是一種文本協(xié)議,所以,它的格式也非常簡(jiǎn)單。
1 HTTP GET請(qǐng)求的格式:
GET /path HTTP/1.1
Header1: Value1
Header2: Value2
Header3: Value3
每個(gè)Header一行一個(gè),換行符是\r\n。
2 HTTP POST請(qǐng)求的格式:
POST /path HTTP/1.1
Header1: Value1
Header2: Value2
Header3: Value3
body data goes here...
3 HTTP響應(yīng)的格式:
200 OK
Header1: Value1
Header2: Value2
Header3: Value3
body data goes here...
HTTP響應(yīng)如果包含body,也是通過(guò)\r\n\r\n來(lái)分隔的。
請(qǐng)?jiān)俅巫⒁猓珺ody的數(shù)據(jù)類型由Content-Type頭來(lái)確定,如果是網(wǎng)頁(yè),Body就是文本,如果是圖片,Body就是圖片的二進(jìn)制數(shù)據(jù)。
當(dāng)存在Content-Encoding時(shí),Body數(shù)據(jù)是被壓縮的,最常見(jiàn)的壓縮方式是gzip,所以,看到Content-Encoding: gzip時(shí),需要將Body數(shù)據(jù)先解壓縮,才能得到真正的數(shù)據(jù)。壓縮的目的在于減少Body的大小,加快網(wǎng)絡(luò)傳輸。
demo:靜態(tài)web服務(wù)器
import socket
import re
import time
def service_client(new_socket):
"""為這個(gè)客戶端服務(wù)"""
# 1.接收瀏覽器發(fā)送過(guò)來(lái)的請(qǐng)求,即http請(qǐng)求
# GET / HTTP/1.1
# --------
request = new_socket.recv(1024).decode('utf-8')
# 判斷客戶端意外斷開(kāi)鏈接返回空字符串
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 += "
404 not found
沒(méi)有發(fā)現(xiàn)所請(qǐng)求資源
"
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準(zhǔn)備發(fā)送給瀏覽器的數(shù)據(jù)---header
response = "HTTP/1.1 200 OK\r\n"
response += "\r\n"
# 2.2準(zhǔn)備發(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():
"""用來(lái)完成整體的控制"""
# 1.創(chuàng)建套接字
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 端口復(fù)用
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.為這個(gè)客戶端服務(wù)
service_client(new_socket)
# 關(guān)閉監(jiān)聽(tīng)套接字
tcp_server_socket.close()
if __name__ == '__main__':
main()
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061
微信掃一掃加我為好友
QQ號(hào)聯(lián)系: 360901061
您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對(duì)您有幫助就好】元

