静态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()

  

posted @   Allen_Hao  阅读(20)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 单元测试从入门到精通
· 上周热点回顾(3.3-3.9)
· winform 绘制太阳,地球,月球 运作规律
点击右上角即可分享
微信分享提示