Web静态服务器多任务版

前文的web静态服务器响应完一个客户端,才能接着处理其它客户端强求;
本文利用线程执行多个客户端请求。

import socket
import threading

# 处理客户端请求
def handle_client_request(new_socket):
    # 接收数据
    recv_data = new_socket.recv(4096)
    # 判断客户端是否发送数据
    if len(recv_data) == 0:
        print('客户端断开连接了')
        # 关闭服务于客户端的套接字
        new_socket.close()
        return

    # 对二进制数据进行解码
    recv_content = recv_data.decode('utf-8')
    print(recv_content)
    # 以空格进行分割
    request_list = recv_content.split(' ', 2)
    # 对请求数据切片获取路径
    request_path = request_list[1]
    print(request_path)
    # 判断返回的是否是根目录,如果是则返回首页数据
    if request_path == '/':
        request_path = '/index.html'
    try:
        # 打开发送内容,此时是以'rb'方式打开,打开内容为二进制
        with open('static' + request_path, 'rb') as file:
            # 读取内容
            file_data = file.read()

    except Exception as e:
        # 请求资源不存在,返回404数据
        # 响应行
        response_line = 'HTTP/1.1 404 NOT FOUND\r\n'
        # 响应头
        response_header = 'Server: PWS/1.1\r\n'
        # 打开发送内容,此时是以'rb'方式打开,打开内容为二进制
        with open('static/error.html', 'rb') as file:
            # 读取内容
            file_data = file.read()
        # 响应体,此时的响应体是二进制
        response_body = file_data
        # 将数据封装成http响应报文格式
        response = (response_line +
                    response_header +
                    '\r\n').encode('utf-8') + response_body

        # 发送响应报文数据
        new_socket.send(response)

    else:
        # 响应行
        response_line = 'HTTP/1.1 200 OK\r\n'
        # 响应头
        response_header = 'Server: PWS/1.1\r\n'
        # 响应体,此时的响应体是二进制
        response_body = file_data
        # 将数据封装成http响应报文格式
        response = (response_line +
                    response_header +
                    '\r\n').encode('utf-8') + response_body

        # 发送响应报文数据
        new_socket.send(response)
    finally:
        # 关闭服务于客户端的套接字
        new_socket.close()

# 程序入口
def main():
    # 创建服务端套接字
    tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    # 设置端口号复用, 程序退出,端口号立即释放
    tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
    # 绑定端口号
    tcp_server_socket.bind(('', 8000))
    # 设置监听
    tcp_server_socket.listen(128)
    # 循环等待接受客户端的连接请求
    while True:
        # 等待接受客户端的连接请求
        new_socket, ip_port = tcp_server_socket.accept()
        # 创建子线程
        sub_thread = threading.Thread(target=handle_client_request, args=(new_socket,))
        # 设置守护主线程,主线程退出,子线程直接销毁
        sub_thread.setDaemon(True)
        sub_thread.start()

if __name__ == '__main__':
    main()

这里注意:前文客户端断开连接时使用的是“continue”, 而这里用的是“return”

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值