Python網(wǎng)絡(luò)編程之HTTP協(xié)議的python應(yīng)用
搭建python自帶靜態(tài)web服務(wù)器
1. 靜態(tài)Web服務(wù)器是什么?
可以為發(fā)出請(qǐng)求的瀏覽器提供靜態(tài)文檔的程序。
平時(shí)我們?yōu)g覽百度新聞數(shù)據(jù)的時(shí)候,每天的新聞數(shù)據(jù)都會(huì)發(fā)生變化,那訪問的這個(gè)頁面就是動(dòng)態(tài)的,而我們開發(fā)的是靜態(tài)的,頁面的數(shù)據(jù)不會(huì)發(fā)生變化。
2. 如何搭建Python自帶的靜態(tài)Web服務(wù)器
搭建Python自帶的靜態(tài)Web服務(wù)器使用 python3 -m http.server 端口號(hào), 效果圖如下:

-m選項(xiàng)說明:
-m表示運(yùn)行包里面的模塊,執(zhí)行這個(gè)命令的時(shí)候,需要進(jìn)入你自己指定靜態(tài)文件的目錄,然后通過瀏覽器就能訪問對(duì)應(yīng)的 html文件了,這樣一個(gè)靜態(tài)的web服務(wù)器就搭建好了。
3. 訪問搭建的靜態(tài)Web服務(wù)器
通過瀏覽器訪問搭建的靜態(tài)Web服務(wù)器,效果圖如下:

4. 查看瀏覽器和搭建的靜態(tài)Web服務(wù)器的通信過程
查看http的通信過程,效果圖如下:

靜態(tài)web服務(wù)器-返回固定頁面數(shù)據(jù)
1. 開發(fā)自己的靜態(tài)Web服務(wù)器
實(shí)現(xiàn)步驟:
- 編寫一個(gè)TCP服務(wù)端程序
- 獲取瀏覽器發(fā)送的http請(qǐng)求報(bào)文數(shù)據(jù)
- 讀取固定頁面數(shù)據(jù),把頁面數(shù)據(jù)組裝成HTTP響應(yīng)報(bào)文數(shù)據(jù)發(fā)送給瀏覽器。
- HTTP響應(yīng)報(bào)文數(shù)據(jù)發(fā)送完成以后,關(guān)閉服務(wù)于客戶端的套接字。
2. 靜態(tài)Web服務(wù)器-返回固定頁面數(shù)據(jù)的示例代碼
import socket
if __name__ == '__main__':
# 創(chuàng)建tcp服務(wù)端套接字
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 設(shè)置端口號(hào)復(fù)用, 程序退出端口立即釋放
tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
# 綁定端口號(hào)
tcp_server_socket.bind(("", 9000))
# 設(shè)置監(jiān)聽
tcp_server_socket.listen(128)
while True:
# 等待接受客戶端的連接請(qǐng)求
new_socket, ip_port = tcp_server_socket.accept()
# 代碼執(zhí)行到此,說明連接建立成功
recv_client_data = new_socket.recv(4096)
# 對(duì)二進(jìn)制數(shù)據(jù)進(jìn)行解碼
recv_client_content = recv_client_data.decode("utf-8")
print(recv_client_content)
with open("static/index.html", "rb") as file:
# 讀取文件數(shù)據(jù)
file_data = file.read()
# 響應(yīng)行
response_line = "HTTP/1.1 200 OK\r\n"
# 響應(yīng)頭
response_header = "Server: PWS1.0\r\n"
# 響應(yīng)體
response_body = file_data
# 拼接響應(yīng)報(bào)文
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body
# 發(fā)送數(shù)據(jù)
new_socket.send(response_data)
# 關(guān)閉服務(wù)與客戶端的套接字
new_socket.close()
靜態(tài)web服務(wù)器-返回指定頁面數(shù)據(jù)
1. 靜態(tài)Web服務(wù)器的問題

目前的Web服務(wù)器,不管用戶訪問什么頁面,返回的都是固定頁面的數(shù)據(jù),接下來需要根據(jù)用戶的請(qǐng)求返回指定頁面的數(shù)據(jù)
返回指定頁面數(shù)據(jù)的實(shí)現(xiàn)步驟:
- 獲取用戶請(qǐng)求資源的路徑
- 根據(jù)請(qǐng)求資源的路徑,讀取指定文件的數(shù)據(jù)
- 組裝指定文件數(shù)據(jù)的響應(yīng)報(bào)文,發(fā)送給瀏覽器
- 判斷請(qǐng)求的文件在服務(wù)端不存在,組裝404狀態(tài)的響應(yīng)報(bào)文,發(fā)送給瀏覽器
2. 靜態(tài)Web服務(wù)器-返回指定頁面數(shù)據(jù)的示例代碼
import socket
def main():
# 創(chuàng)建tcp服務(wù)端套接字
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 設(shè)置端口號(hào)復(fù)用, 程序退出端口立即釋放
tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
# 綁定端口號(hào)
tcp_server_socket.bind(("", 9000))
# 設(shè)置監(jiān)聽
tcp_server_socket.listen(128)
while True:
# 等待接受客戶端的連接請(qǐng)求
new_socket, ip_port = tcp_server_socket.accept()
# 代碼執(zhí)行到此,說明連接建立成功
recv_client_data = new_socket.recv(4096)
if len(recv_client_data) == 0:
print("關(guān)閉瀏覽器了")
new_socket.close()
return
# 對(duì)二進(jìn)制數(shù)據(jù)進(jìn)行解碼
recv_client_content = recv_client_data.decode("utf-8")
print(recv_client_content)
# 根據(jù)指定字符串進(jìn)行分割, 最大分割次數(shù)指定2
request_list = recv_client_content.split(" ", maxsplit=2)
# 獲取請(qǐng)求資源路徑
request_path = request_list[1]
print(request_path)
# 判斷請(qǐng)求的是否是根目錄,如果條件成立,指定首頁數(shù)據(jù)返回
if request_path == "/":
request_path = "/index.html"
try:
# 動(dòng)態(tài)打開指定文件
with open("static" + request_path, "rb") as file:
# 讀取文件數(shù)據(jù)
file_data = file.read()
except Exception as e:
# 請(qǐng)求資源不存在,返回404數(shù)據(jù)
# 響應(yīng)行
response_line = "HTTP/1.1 404 Not Found\r\n"
# 響應(yīng)頭
response_header = "Server: PWS1.0\r\n"
with open("static/error.html", "rb") as file:
file_data = file.read()
# 響應(yīng)體
response_body = file_data
# 拼接響應(yīng)報(bào)文
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body
# 發(fā)送數(shù)據(jù)
new_socket.send(response_data)
else:
# 響應(yīng)行
response_line = "HTTP/1.1 200 OK\r\n"
# 響應(yīng)頭
response_header = "Server: PWS1.0\r\n"
# 響應(yīng)體
response_body = file_data
# 拼接響應(yīng)報(bào)文
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body
# 發(fā)送數(shù)據(jù)
new_socket.send(response_data)
finally:
# 關(guān)閉服務(wù)與客戶端的套接字
new_socket.close()
if __name__ == '__main__':
main()
靜態(tài)web服務(wù)器-多任務(wù)實(shí)現(xiàn)
1. 靜態(tài)Web服務(wù)器的問題
目前的Web服務(wù)器,不能支持多用戶同時(shí)訪問,只能一個(gè)一個(gè)的處理客戶端的請(qǐng)求,那么如何開發(fā)多任務(wù)版的web服務(wù)器同時(shí)處理 多個(gè)客戶端的請(qǐng)求?
可以使用多線程,比進(jìn)程更加節(jié)省內(nèi)存資源。
多任務(wù)版web服務(wù)器程序的實(shí)現(xiàn)步驟:
當(dāng)客戶端和服務(wù)端建立連接成功,創(chuàng)建子線程,使用子線程專門處理客戶端的請(qǐng)求,防止主線程阻塞。
把創(chuàng)建的子線程設(shè)置成為守護(hù)主線程,防止主線程無法退出。
把創(chuàng)建的子線程設(shè)置成為守護(hù)主線程,防止主線程無法退出。
2. 靜態(tài)Web服務(wù)器-多任務(wù)版的示例代碼
import socket
import threading
# 處理客戶端的請(qǐng)求
def handle_client_request(new_socket):
# 代碼執(zhí)行到此,說明連接建立成功
recv_client_data = new_socket.recv(4096)
if len(recv_client_data) == 0:
print("關(guān)閉瀏覽器了")
new_socket.close()
return
# 對(duì)二進(jìn)制數(shù)據(jù)進(jìn)行解碼
recv_client_content = recv_client_data.decode("utf-8")
print(recv_client_content)
# 根據(jù)指定字符串進(jìn)行分割, 最大分割次數(shù)指定2
request_list = recv_client_content.split(" ", maxsplit=2)
# 獲取請(qǐng)求資源路徑
request_path = request_list[1]
print(request_path)
# 判斷請(qǐng)求的是否是根目錄,如果條件成立,指定首頁數(shù)據(jù)返回
if request_path == "/":
request_path = "/index.html"
try:
# 動(dòng)態(tài)打開指定文件
with open("static" + request_path, "rb") as file:
# 讀取文件數(shù)據(jù)
file_data = file.read()
except Exception as e:
# 請(qǐng)求資源不存在,返回404數(shù)據(jù)
# 響應(yīng)行
response_line = "HTTP/1.1 404 Not Found\r\n"
# 響應(yīng)頭
response_header = "Server: PWS1.0\r\n"
with open("static/error.html", "rb") as file:
file_data = file.read()
# 響應(yīng)體
response_body = file_data
# 拼接響應(yīng)報(bào)文
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body
# 發(fā)送數(shù)據(jù)
new_socket.send(response_data)
else:
# 響應(yīng)行
response_line = "HTTP/1.1 200 OK\r\n"
# 響應(yīng)頭
response_header = "Server: PWS1.0\r\n"
# 響應(yīng)體
response_body = file_data
# 拼接響應(yīng)報(bào)文
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body
# 發(fā)送數(shù)據(jù)
new_socket.send(response_data)
finally:
# 關(guān)閉服務(wù)與客戶端的套接字
new_socket.close()
# 程序入口函數(shù)
def main():
# 創(chuàng)建tcp服務(wù)端套接字
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 設(shè)置端口號(hào)復(fù)用, 程序退出端口立即釋放
tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
# 綁定端口號(hào)
tcp_server_socket.bind(("", 9000))
# 設(shè)置監(jiān)聽
tcp_server_socket.listen(128)
while True:
# 等待接受客戶端的連接請(qǐng)求
new_socket, ip_port = tcp_server_socket.accept()
print(ip_port)
# 當(dāng)客戶端和服務(wù)器建立連接程,創(chuàng)建子線程
sub_thread = threading.Thread(target=handle_client_request, args=(new_socket,))
# 設(shè)置守護(hù)主線程
sub_thread.setDaemon(True)
# 啟動(dòng)子線程執(zhí)行對(duì)應(yīng)的任務(wù)
sub_thread.start()
if __name__ == '__main__':
main()
靜態(tài)web服務(wù)器-面向?qū)ο箝_發(fā)
1. 以面向?qū)ο蟮姆绞介_發(fā)靜態(tài)Web服務(wù)器
實(shí)現(xiàn)步驟:
- 把提供服務(wù)的Web服務(wù)器抽象成一個(gè)類(HTTPWebServer)
- 提供Web服務(wù)器的初始化方法,在初始化方法里面創(chuàng)建socket對(duì)象
- 提供一個(gè)開啟Web服務(wù)器的方法,讓W(xué)eb服務(wù)器處理客戶端請(qǐng)求操作。
2. 靜態(tài)Web服務(wù)器-面向?qū)ο箝_發(fā)的示例代碼
import socket
import threading
# 定義web服務(wù)器類
class HttpWebServer(object):
def __init__(self):
# 創(chuàng)建tcp服務(wù)端套接字
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 設(shè)置端口號(hào)復(fù)用, 程序退出端口立即釋放
tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
# 綁定端口號(hào)
tcp_server_socket.bind(("", 9000))
# 設(shè)置監(jiān)聽
tcp_server_socket.listen(128)
# 保存創(chuàng)建成功的服務(wù)器套接字
self.tcp_server_socket = tcp_server_socket
# 處理客戶端的請(qǐng)求
@staticmethod
def handle_client_request(new_socket):
# 代碼執(zhí)行到此,說明連接建立成功
recv_client_data = new_socket.recv(4096)
if len(recv_client_data) == 0:
print("關(guān)閉瀏覽器了")
new_socket.close()
return
# 對(duì)二進(jìn)制數(shù)據(jù)進(jìn)行解碼
recv_client_content = recv_client_data.decode("utf-8")
print(recv_client_content)
# 根據(jù)指定字符串進(jìn)行分割, 最大分割次數(shù)指定2
request_list = recv_client_content.split(" ", maxsplit=2)
# 獲取請(qǐng)求資源路徑
request_path = request_list[1]
print(request_path)
# 判斷請(qǐng)求的是否是根目錄,如果條件成立,指定首頁數(shù)據(jù)返回
if request_path == "/":
request_path = "/index.html"
try:
# 動(dòng)態(tài)打開指定文件
with open("static" + request_path, "rb") as file:
# 讀取文件數(shù)據(jù)
file_data = file.read()
except Exception as e:
# 請(qǐng)求資源不存在,返回404數(shù)據(jù)
# 響應(yīng)行
response_line = "HTTP/1.1 404 Not Found\r\n"
# 響應(yīng)頭
response_header = "Server: PWS1.0\r\n"
with open("static/error.html", "rb") as file:
file_data = file.read()
# 響應(yīng)體
response_body = file_data
# 拼接響應(yīng)報(bào)文
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body
# 發(fā)送數(shù)據(jù)
new_socket.send(response_data)
else:
# 響應(yīng)行
response_line = "HTTP/1.1 200 OK\r\n"
# 響應(yīng)頭
response_header = "Server: PWS1.0\r\n"
# 響應(yīng)體
response_body = file_data
# 拼接響應(yīng)報(bào)文
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body
# 發(fā)送數(shù)據(jù)
new_socket.send(response_data)
finally:
# 關(guān)閉服務(wù)與客戶端的套接字
new_socket.close()
# 啟動(dòng)web服務(wù)器進(jìn)行工作
def start(self):
while True:
# 等待接受客戶端的連接請(qǐng)求
new_socket, ip_port = self.tcp_server_socket.accept()
# 當(dāng)客戶端和服務(wù)器建立連接程,創(chuàng)建子線程
sub_thread = threading.Thread(target=self.handle_client_request, args=(new_socket,))
# 設(shè)置守護(hù)主線程
sub_thread.setDaemon(True)
# 啟動(dòng)子線程執(zhí)行對(duì)應(yīng)的任務(wù)
sub_thread.start()
# 程序入口函數(shù)
def main():
# 創(chuàng)建web服務(wù)器對(duì)象
web_server = HttpWebServer()
# 啟動(dòng)web服務(wù)器進(jìn)行工作
web_server.start()
if __name__ == '__main__':
main()
靜態(tài)web服務(wù)器-命令行啟動(dòng)動(dòng)態(tài)綁定端口號(hào)
1. 開發(fā)命令行啟動(dòng)動(dòng)態(tài)綁定端口號(hào)的靜態(tài)web服務(wù)器
實(shí)現(xiàn)步驟:
- 獲取執(zhí)行python程序的終端命令行參數(shù)
- 判斷參數(shù)的類型,設(shè)置端口號(hào)必須是整型
- 給Web服務(wù)器類的初始化方法添加一個(gè)端口號(hào)參數(shù),用于綁定端口號(hào)
2. 靜態(tài)Web服務(wù)器-命令行啟動(dòng)動(dòng)態(tài)綁定端口號(hào)的示例代碼
import socket
import threading
import sys
# 定義web服務(wù)器類
class HttpWebServer(object):
def __init__(self, port):
# 創(chuàng)建tcp服務(wù)端套接字
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 設(shè)置端口號(hào)復(fù)用, 程序退出端口立即釋放
tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
# 綁定端口號(hào)
tcp_server_socket.bind(("", port))
# 設(shè)置監(jiān)聽
tcp_server_socket.listen(128)
# 保存創(chuàng)建成功的服務(wù)器套接字
self.tcp_server_socket = tcp_server_socket
# 處理客戶端的請(qǐng)求
@staticmethod
def handle_client_request(new_socket):
# 代碼執(zhí)行到此,說明連接建立成功
recv_client_data = new_socket.recv(4096)
if len(recv_client_data) == 0:
print("關(guān)閉瀏覽器了")
new_socket.close()
return
# 對(duì)二進(jìn)制數(shù)據(jù)進(jìn)行解碼
recv_client_content = recv_client_data.decode("utf-8")
print(recv_client_content)
# 根據(jù)指定字符串進(jìn)行分割, 最大分割次數(shù)指定2
request_list = recv_client_content.split(" ", maxsplit=2)
# 獲取請(qǐng)求資源路徑
request_path = request_list[1]
print(request_path)
# 判斷請(qǐng)求的是否是根目錄,如果條件成立,指定首頁數(shù)據(jù)返回
if request_path == "/":
request_path = "/index.html"
try:
# 動(dòng)態(tài)打開指定文件
with open("static" + request_path, "rb") as file:
# 讀取文件數(shù)據(jù)
file_data = file.read()
except Exception as e:
# 請(qǐng)求資源不存在,返回404數(shù)據(jù)
# 響應(yīng)行
response_line = "HTTP/1.1 404 Not Found\r\n"
# 響應(yīng)頭
response_header = "Server: PWS1.0\r\n"
with open("static/error.html", "rb") as file:
file_data = file.read()
# 響應(yīng)體
response_body = file_data
# 拼接響應(yīng)報(bào)文
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body
# 發(fā)送數(shù)據(jù)
new_socket.send(response_data)
else:
# 響應(yīng)行
response_line = "HTTP/1.1 200 OK\r\n"
# 響應(yīng)頭
response_header = "Server: PWS1.0\r\n"
# 響應(yīng)體
response_body = file_data
# 拼接響應(yīng)報(bào)文
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body
# 發(fā)送數(shù)據(jù)
new_socket.send(response_data)
finally:
# 關(guān)閉服務(wù)與客戶端的套接字
new_socket.close()
# 啟動(dòng)web服務(wù)器進(jìn)行工作
def start(self):
while True:
# 等待接受客戶端的連接請(qǐng)求
new_socket, ip_port = self.tcp_server_socket.accept()
# 當(dāng)客戶端和服務(wù)器建立連接程,創(chuàng)建子線程
sub_thread = threading.Thread(target=self.handle_client_request, args=(new_socket,))
# 設(shè)置守護(hù)主線程
sub_thread.setDaemon(True)
# 啟動(dòng)子線程執(zhí)行對(duì)應(yīng)的任務(wù)
sub_thread.start()
# 程序入口函數(shù)
def main():
print(sys.argv)
# 判斷命令行參數(shù)是否等于2,
if len(sys.argv) != 2:
print("執(zhí)行命令如下: python3 xxx.py 8000")
return
# 判斷字符串是否都是數(shù)字組成
if not sys.argv[1].isdigit():
print("執(zhí)行命令如下: python3 xxx.py 8000")
return
# 獲取終端命令行參數(shù)
port = int(sys.argv[1])
# 創(chuàng)建web服務(wù)器對(duì)象
web_server = HttpWebServer(port)
# 啟動(dòng)web服務(wù)器進(jìn)行工作
web_server.start()
if __name__ == '__main__':
main()
到此這篇關(guān)于Python網(wǎng)絡(luò)編程之HTTP協(xié)議的python應(yīng)用的文章就介紹到這了,更多相關(guān)python http協(xié)議內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python命令行定時(shí)任務(wù)自動(dòng)化工作流程
本文介紹如何使用Python編寫定時(shí)任務(wù),以自動(dòng)執(zhí)行命令行任務(wù)。您將學(xué)習(xí)如何安排定期的任務(wù),處理任務(wù)結(jié)果,以及如何使用Python自動(dòng)化工作流程,從而提高工作效率。無需手動(dòng)執(zhí)行重復(fù)任務(wù),Python幫您搞定2023-04-04
python被修飾的函數(shù)消失問題解決(基于wraps函數(shù))
這篇文章主要介紹了python被修飾的函數(shù)消失問題解決(基于wraps函數(shù)),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11
pytorch載入預(yù)訓(xùn)練模型后,實(shí)現(xiàn)訓(xùn)練指定層
今天小編就為大家分享一篇pytorch載入預(yù)訓(xùn)練模型后,實(shí)現(xiàn)訓(xùn)練指定層,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-01-01
python?argparse的使用步驟(全網(wǎng)最全)
argparse是python的一個(gè)命令行參數(shù)解析包,在代碼需要頻繁修改參數(shù)時(shí),方便使用,主要用法就是在命令行輸入自己想要修改的參數(shù),這篇文章主要介紹了python?argparse的使用步驟(全網(wǎng)最全),需要的朋友可以參考下2023-04-04
Python操控Chrome瀏覽器進(jìn)行網(wǎng)頁操作
這篇文章將為您展示如何通過Python控制瀏覽器實(shí)現(xiàn)網(wǎng)頁的打開、頁面的切換和關(guān)閉的基本操作,文中的示例代碼講解詳細(xì),感興趣的可以了解一下2023-06-06

