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ā)生變化,那訪問(wèn)的這個(gè)頁(yè)面就是動(dòng)態(tài)的,而我們開(kāi)發(fā)的是靜態(tài)的,頁(yè)面的數(shù)據(jù)不會(huì)發(fā)生變化。
2. 如何搭建Python自帶的靜態(tài)Web服務(wù)器
搭建Python自帶的靜態(tài)Web服務(wù)器使用 python3 -m http.server 端口號(hào), 效果圖如下:
-m選項(xiàng)說(shuō)明:
-m表示運(yùn)行包里面的模塊,執(zhí)行這個(gè)命令的時(shí)候,需要進(jìn)入你自己指定靜態(tài)文件的目錄,然后通過(guò)瀏覽器就能訪問(wèn)對(duì)應(yīng)的 html文件了,這樣一個(gè)靜態(tài)的web服務(wù)器就搭建好了。
3. 訪問(wèn)搭建的靜態(tài)Web服務(wù)器
通過(guò)瀏覽器訪問(wèn)搭建的靜態(tài)Web服務(wù)器,效果圖如下:
4. 查看瀏覽器和搭建的靜態(tài)Web服務(wù)器的通信過(guò)程
查看http的通信過(guò)程,效果圖如下:
靜態(tài)web服務(wù)器-返回固定頁(yè)面數(shù)據(jù)
1. 開(kāi)發(fā)自己的靜態(tài)Web服務(wù)器
實(shí)現(xiàn)步驟:
- 編寫(xiě)一個(gè)TCP服務(wù)端程序
- 獲取瀏覽器發(fā)送的http請(qǐng)求報(bào)文數(shù)據(jù)
- 讀取固定頁(yè)面數(shù)據(jù),把頁(yè)面數(shù)據(jù)組裝成HTTP響應(yīng)報(bào)文數(shù)據(jù)發(fā)送給瀏覽器。
- HTTP響應(yīng)報(bào)文數(shù)據(jù)發(fā)送完成以后,關(guān)閉服務(wù)于客戶(hù)端的套接字。
2. 靜態(tài)Web服務(wù)器-返回固定頁(yè)面數(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)聽(tīng) tcp_server_socket.listen(128) while True: # 等待接受客戶(hù)端的連接請(qǐng)求 new_socket, ip_port = tcp_server_socket.accept() # 代碼執(zhí)行到此,說(shuō)明連接建立成功 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ù)與客戶(hù)端的套接字 new_socket.close()
靜態(tài)web服務(wù)器-返回指定頁(yè)面數(shù)據(jù)
1. 靜態(tài)Web服務(wù)器的問(wèn)題
目前的Web服務(wù)器,不管用戶(hù)訪問(wèn)什么頁(yè)面,返回的都是固定頁(yè)面的數(shù)據(jù),接下來(lái)需要根據(jù)用戶(hù)的請(qǐng)求返回指定頁(yè)面的數(shù)據(jù)
返回指定頁(yè)面數(shù)據(jù)的實(shí)現(xiàn)步驟:
- 獲取用戶(hù)請(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ù)器-返回指定頁(yè)面數(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)聽(tīng) tcp_server_socket.listen(128) while True: # 等待接受客戶(hù)端的連接請(qǐng)求 new_socket, ip_port = tcp_server_socket.accept() # 代碼執(zhí)行到此,說(shuō)明連接建立成功 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)求的是否是根目錄,如果條件成立,指定首頁(yè)數(shù)據(jù)返回 if request_path == "/": request_path = "/index.html" try: # 動(dòng)態(tài)打開(kā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ù)與客戶(hù)端的套接字 new_socket.close() if __name__ == '__main__': main()
靜態(tài)web服務(wù)器-多任務(wù)實(shí)現(xiàn)
1. 靜態(tài)Web服務(wù)器的問(wèn)題
目前的Web服務(wù)器,不能支持多用戶(hù)同時(shí)訪問(wèn),只能一個(gè)一個(gè)的處理客戶(hù)端的請(qǐng)求,那么如何開(kāi)發(fā)多任務(wù)版的web服務(wù)器同時(shí)處理 多個(gè)客戶(hù)端的請(qǐng)求?
可以使用多線程,比進(jìn)程更加節(jié)省內(nèi)存資源。
多任務(wù)版web服務(wù)器程序的實(shí)現(xiàn)步驟:
當(dāng)客戶(hù)端和服務(wù)端建立連接成功,創(chuàng)建子線程,使用子線程專(zhuān)門(mén)處理客戶(hù)端的請(qǐng)求,防止主線程阻塞。
把創(chuàng)建的子線程設(shè)置成為守護(hù)主線程,防止主線程無(wú)法退出。
把創(chuàng)建的子線程設(shè)置成為守護(hù)主線程,防止主線程無(wú)法退出。
2. 靜態(tài)Web服務(wù)器-多任務(wù)版的示例代碼
import socket import threading # 處理客戶(hù)端的請(qǐng)求 def handle_client_request(new_socket): # 代碼執(zhí)行到此,說(shuō)明連接建立成功 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)求的是否是根目錄,如果條件成立,指定首頁(yè)數(shù)據(jù)返回 if request_path == "/": request_path = "/index.html" try: # 動(dòng)態(tài)打開(kā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ù)與客戶(hù)端的套接字 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)聽(tīng) tcp_server_socket.listen(128) while True: # 等待接受客戶(hù)端的連接請(qǐng)求 new_socket, ip_port = tcp_server_socket.accept() print(ip_port) # 當(dāng)客戶(hù)端和服務(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ū)ο箝_(kāi)發(fā)
1. 以面向?qū)ο蟮姆绞介_(kāi)發(fā)靜態(tài)Web服務(wù)器
實(shí)現(xiàn)步驟:
- 把提供服務(wù)的Web服務(wù)器抽象成一個(gè)類(lèi)(HTTPWebServer)
- 提供Web服務(wù)器的初始化方法,在初始化方法里面創(chuàng)建socket對(duì)象
- 提供一個(gè)開(kāi)啟Web服務(wù)器的方法,讓W(xué)eb服務(wù)器處理客戶(hù)端請(qǐng)求操作。
2. 靜態(tài)Web服務(wù)器-面向?qū)ο箝_(kāi)發(fā)的示例代碼
import socket import threading # 定義web服務(wù)器類(lèi) 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)聽(tīng) tcp_server_socket.listen(128) # 保存創(chuàng)建成功的服務(wù)器套接字 self.tcp_server_socket = tcp_server_socket # 處理客戶(hù)端的請(qǐng)求 @staticmethod def handle_client_request(new_socket): # 代碼執(zhí)行到此,說(shuō)明連接建立成功 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)求的是否是根目錄,如果條件成立,指定首頁(yè)數(shù)據(jù)返回 if request_path == "/": request_path = "/index.html" try: # 動(dòng)態(tài)打開(kā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ù)與客戶(hù)端的套接字 new_socket.close() # 啟動(dòng)web服務(wù)器進(jìn)行工作 def start(self): while True: # 等待接受客戶(hù)端的連接請(qǐng)求 new_socket, ip_port = self.tcp_server_socket.accept() # 當(dāng)客戶(hù)端和服務(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. 開(kāi)發(fā)命令行啟動(dòng)動(dòng)態(tài)綁定端口號(hào)的靜態(tài)web服務(wù)器
實(shí)現(xiàn)步驟:
- 獲取執(zhí)行python程序的終端命令行參數(shù)
- 判斷參數(shù)的類(lèi)型,設(shè)置端口號(hào)必須是整型
- 給Web服務(wù)器類(lèi)的初始化方法添加一個(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ù)器類(lèi) 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)聽(tīng) tcp_server_socket.listen(128) # 保存創(chuàng)建成功的服務(wù)器套接字 self.tcp_server_socket = tcp_server_socket # 處理客戶(hù)端的請(qǐng)求 @staticmethod def handle_client_request(new_socket): # 代碼執(zhí)行到此,說(shuō)明連接建立成功 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)求的是否是根目錄,如果條件成立,指定首頁(yè)數(shù)據(jù)返回 if request_path == "/": request_path = "/index.html" try: # 動(dòng)態(tài)打開(kā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ù)與客戶(hù)端的套接字 new_socket.close() # 啟動(dòng)web服務(wù)器進(jìn)行工作 def start(self): while True: # 等待接受客戶(hù)端的連接請(qǐng)求 new_socket, ip_port = self.tcp_server_socket.accept() # 當(dāng)客戶(hù)端和服務(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編寫(xiě)定時(shí)任務(wù),以自動(dòng)執(zhí)行命令行任務(wù)。您將學(xué)習(xí)如何安排定期的任務(wù),處理任務(wù)結(jié)果,以及如何使用Python自動(dòng)化工作流程,從而提高工作效率。無(wú)需手動(dòng)執(zhí)行重復(fù)任務(wù),Python幫您搞定2023-04-04python和c語(yǔ)言哪個(gè)更適合初學(xué)者
在本篇文章里小編給大家分享的是一篇關(guān)于python和c語(yǔ)言哪個(gè)更適合初學(xué)者的相關(guān)文章,有興趣的朋友們學(xué)習(xí)下。2020-06-06python爬蟲(chóng)通過(guò)增加多線程獲取數(shù)據(jù)
這篇文章主要為大家介紹了python爬蟲(chóng)通過(guò)增加多線程獲取數(shù)據(jù)實(shí)現(xiàn)過(guò)程解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06python被修飾的函數(shù)消失問(wèn)題解決(基于wraps函數(shù))
這篇文章主要介紹了python被修飾的函數(shù)消失問(wèn)題解決(基于wraps函數(shù)),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11pytorch載入預(yù)訓(xùn)練模型后,實(shí)現(xiàn)訓(xùn)練指定層
今天小編就為大家分享一篇pytorch載入預(yù)訓(xùn)練模型后,實(shí)現(xiàn)訓(xùn)練指定層,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-01-01python?argparse的使用步驟(全網(wǎng)最全)
argparse是python的一個(gè)命令行參數(shù)解析包,在代碼需要頻繁修改參數(shù)時(shí),方便使用,主要用法就是在命令行輸入自己想要修改的參數(shù),這篇文章主要介紹了python?argparse的使用步驟(全網(wǎng)最全),需要的朋友可以參考下2023-04-04Python操控Chrome瀏覽器進(jìn)行網(wǎng)頁(yè)操作
這篇文章將為您展示如何通過(guò)Python控制瀏覽器實(shí)現(xiàn)網(wǎng)頁(yè)的打開(kāi)、頁(yè)面的切換和關(guān)閉的基本操作,文中的示例代碼講解詳細(xì),感興趣的可以了解一下2023-06-06