静态Web服务器-多任务版
静态Web服务器的问题
目前的Web服务器,不能支持多用户同时访问,只能一个一个的处理客户端的请求,那么如何开发多任务版的web服务器同时处理多个客户端的请求?
可以使用多线程,比进程更加节省内存资源。
多任务版web服务器程序的实现步骤
1. 当客户端和服务端建⽴连接成功,创建⼦线程,使⽤⼦线程专⻔处理客户端的请求,防⽌主线程阻塞。
2. 把创建的⼦线程设置成为守护主线程,防⽌主线程⽆法退出。
示例
import socket import threading # 获取用户请求资源的路径 # 根据请求资源的路径,读取指定文件的数据 # 组装指定文件数据的响应报文,发送给浏览器 # 判断请求的文件在服务端不存在,组装404状态的响应报文,发送给浏览器 def handle_client_request(client_socekt): # 获取浏览器的请求信息 client_request_data = client_socekt.recv(1024).decode() print(client_request_data) # 获取用户请求资源的路径 requst_data = client_request_data.split(" ") print(requst_data) # 判断客户端是否关闭 if len(requst_data) == 1: client_socekt.close() return # 求资源的路径 request_path = requst_data[1] if request_path == "/": request_path = "/index.html" # 3.读取固定页面数据,把页面数据组装成HTTP响应报文数据发送给浏览器 # 根据请求资源的路径,读取指定文件的数据 try: with open("./static" + request_path, "rb") as f: file_data = f.read() except Exception as e: # 返回404错误数据 # 应答行 response_line = "HTTP/1.1 404 Not Found\r\n" # 应答头 response_header = "Server:pwb\r\n" # 应答体 response_body = "404 Not Found sorry" # 应答数据 # 组装指定文件数据的响应报文,发送给浏览器 response_data = (response_line + response_header + "\r\n" + response_body).encode() client_socekt.send(response_data) else: # 应答行 response_line = "HTTP/1.1 200 OK\r\n" # 应答头 response_header = "Server:pwb\r\n" # 应答体 response_body = file_data # 应答数据 # 组装指定文件数据的响应报文,发送给浏览器 response_data = (response_line + response_header + "\r\n").encode() + response_body client_socekt.send(response_data) finally: # 4.HTTP响应报文数据发送完成以后,关闭服务于客户端的套接字 client_socekt.close() if __name__ == '__main__': # 1.编写一个TCP服务端程序 # 创建socekt tcp_server_socekt = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 设置端口复用 tcp_server_socekt.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True) # 绑定地址 tcp_server_socekt.bind(("", 8080)) # 设置监听 tcp_server_socekt.listen(128) while True: # 2.获取浏览器发送的HTTP请求报文数据 # 建立链接 client_socekt, client_addr = tcp_server_socekt.accept() # 创建子线程 sub_thread = threading.Thread(target=handle_client_request, args=(client_socekt,)) sub_thread.start()
服务端控制台打印:
GET /test.css HTTP/1.1
Host: localhost:8080
Connection: keep-alive
Cache-Control: max-age=0
sec-ch-ua: "Chromium";v="118", "Google Chrome";v="118", "Not=A?Brand";v="99"
sec-ch-ua-mobile: ?0
sec-ch-ua-platform: "Windows"
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7
Sec-Fetch-Site: none
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Sec-Fetch-Dest: document
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9
Cookie: Idea-30eca14b=5671be82-bc4b-4154-8aa0-7ea8dc722c2d; Pycharm-a2976cb7=daed6ffd-95bc-4940-b9b4-2562e9a6fdca
['GET', '/test.css', 'HTTP/1.1\r\nHost:', 'localhost:8080\r\nConnection:', 'keep-alive\r\nCache-Control:', 'max-age=0\r\nsec-ch-ua:', '"Chromium";v="118",', '"Google', 'Chrome";v="118",', '"Not=A?Brand";v="99"\r\nsec-ch-ua-mobile:', '?0\r\nsec-ch-ua-platform:', '"Windows"\r\nUpgrade-Insecure-Requests:', '1\r\nUser-Agent:', 'Mozilla/5.0', '(Windows', 'NT', '10.0;', 'Win64;', 'x64)', 'AppleWebKit/537.36', '(KHTML,', 'like', 'Gecko)', 'Chrome/118.0.0.0', 'Safari/537.36\r\nAccept:', 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7\r\nSec-Fetch-Site:', 'none\r\nSec-Fetch-Mode:', 'navigate\r\nSec-Fetch-User:', '?1\r\nSec-Fetch-Dest:', 'document\r\nAccept-Encoding:', 'gzip,', 'deflate,', 'br\r\nAccept-Language:', 'zh-CN,zh;q=0.9\r\nCookie:', 'Idea-30eca14b=5671be82-bc4b-4154-8aa0-7ea8dc722c2d;', 'Pycharm-a2976cb7=daed6ffd-95bc-4940-b9b4-2562e9a6fdca\r\n\r\n']
['']
['']
小结
1 2 3 4 5 6 7 8 9 10 11 | 1. 当客户端和服务端建立连接成功,创建子线程,使用子线程专门处理客户端的请求,防止主线程阻塞。 while True : # 2.获取浏览器发送的HTTP请求报文数据 # 建立链接 client_socekt, client_addr = tcp_server_socekt.accept() # 创建子线程 sub_thread = threading.Thread(target = handle_client_request, args = (client_socekt,)) 2. 把创建的子线程设置成为守护主线程,防止主线程无法退出。 sub_thread = threading.Thread(target = handle_client_request, args = (client_socekt,)) sub_thread.setDaemon( True ) sub_thread.start() |
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 单元测试从入门到精通
· 上周热点回顾(3.3-3.9)
· winform 绘制太阳,地球,月球 运作规律