zoukankan      html  css  js  c++  java
  • 【Python高级编程038 ● 静态web服务器 ● 静态Web服务器-面向对象版】


    ---------Python基础编程---------

    Author : AI菌


    【内容讲解】

    以面向对象的方式开发静态Web服务器实现步骤:
        1、把提供服务的Web服务器抽象成一个类(HTTPWebServer)
        2、提供Web服务器的初始化方法__init__,在初始化方法里面创建socket对象
        3、提供一个开启Web服务器的方法,让Web服务器处理客户端请求操作

    【代码演示】

    """
    以面向对象的方式开发静态Web服务器实现步骤:
        1、把提供服务的Web服务器抽象成一个类(HTTPWebServer)
        2、提供Web服务器的初始化方法__init__,在初始化方法里面创建socket对象
        3、提供一个开启Web服务器的方法,让Web服务器处理客户端请求操作
    """
    
    import socket
    import threading
    
    
    # 1、定义web服务器类
    class HttpWebServer(object):
        # 2、提供Web服务器的初始化方法__init__,在初始化方法里面创建socket对象
        def __init__(self):
            # 创建tcp服务端套接字
            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)
            # 保存创建好的套接字
            self.tcp_server_socket = tcp_server_socket
    
        # 处理客户端的请求
        # handle_client_request方法没有用到self,可以定义为静态方法
        @staticmethod
        def handle_client_request(new_socket):
            # 接收客户端的数据
            recv_client_data = new_socket.recv(4096)
    
            # 判断接收的客户端数据是否为0
            if len(recv_client_data) == 0:
                new_socket.close()
                return
    
            # 对二进制数据进行解码
            recv_client_content = recv_client_data.decode("utf-8")
            print("来自客户端的数据:", recv_client_content)
    
            # 对数据按照空格进行分割,分割2次
            request_list = recv_client_content.split(" ", maxsplit=2)
            # 获取请求资源的路径
            request_path = request_list[1]
            print("截取的请求资源的路径:", request_path)
    
            # 判断请求的是否是根目录,如果是根目录设置返回的信息
            if request_path == "/":
                request_path = "/index.html"
    
            try:
                # 读取文件数据
                # rb表示以二进制的方式读取文件数据,图片数据需要以二进制格式打开,使用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
    "
                # 响应头
                response_header = "server:DiamondDownload1.0
    "
                # 读取404页面数据
                with open("static/error.html", "rb") as file:
                    file_data = file.read()
                # 响应体
                response_body = file_data
                # 把数据封装成http响应报文格式的数据
                response_data = (response_line + response_header + "
    ").encode("utf-8") + response_body
    
                # 发送给浏览器的响应报文数据
                new_socket.send(response_data)
            else:
                # 代码执行到此,说明有请求的文件,返回200状态信息
                # 响应行
                response_line = "HTTP/1.1 200 OK
    "
                # 响应头
                response_header = "server:DiamondDownload1.0
    "
                # 响应体
                response_body = file_data
                # 把数据封装成http响应报文格式的数据
                response_data = (response_line + response_header + "
    ").encode("utf-8") + response_body
    
                # 发送给浏览器的响应报文数据
                new_socket.send(response_data)
            finally:
                # 关闭服务端与客户端套接字
                new_socket.close()
    
        # 3、提供一个开启Web服务器的方法,让Web服务器处理客户端请求操作
        def start(self):
            # 循环响应客户端请求
            while True:
                # 等待接受客户端的连接请求
                new_socket, port = self.tcp_server_socket.accept()
                # 代码执行到此,说明连接建立成功
    
                # 当客户端和服务器建立连接程,创建子线程
                sub_thread = threading.Thread(target=self.handle_client_request, args=(new_socket,))
                # 设置守护线程
                sub_thread.setDaemon(True)
                # 启动子线程执行对应的任务
                sub_thread.start()
    
    
    # 程序入口函数
    def main():
        # 创建web服务器对象
        web_server = HttpWebServer()
        # 启动web服务器进行工作
        web_server.start()
    
    
    if __name__ == '__main__':
        main()

    【运行结果】

     服务端程序控制台打印结果:

    来自客户端的数据: GET /index.html HTTP/1.1
    Host: 192.168.1.64:8000
    Connection: keep-alive
    Upgrade-Insecure-Requests: 1
    User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/85.0.4183.102 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.9
    Accept-Encoding: gzip, deflate
    Accept-Language: zh-CN,zh;q=0.9
    
    
    截取的请求资源的路径: /index.html

    浏览器访问:

    【往期精彩】

    ▷【Python基础编程196 ● 读取文件的4种方式】
    ▷【Python基础编程197 ● 读取文件的4种方式】
    ▷【Python基础编程198 ● 读取文件的4种方式】
    ▷【Python基础编程199 ● Python怎么读/写很大的文件】
    ▷【Python基础编程200 ● 读取文件的4种方式】
    ▷【Python基础编程201 ● 读取文件的4种方式】
    ▷【Python基础编程202 ● 读取文件的4种方式】
    ▷【Python基础编程203 ● 读取文件的4种方式】

    【加群交流】



  • 相关阅读:
    xgqfrms™, xgqfrms® : xgqfrms's offical website of GitHub!
    xgqfrms™, xgqfrms® : xgqfrms's offical website of GitHub!
    vue element 表单验证不通过,滚动到固对应位置
    vue源码之数据驱动
    vue源码之数据驱动
    vue源码之数据驱动
    每天一点点之数据结构与算法
    每天一点点之数据结构与算法
    python案例
    python案例
  • 原文地址:https://www.cnblogs.com/hezhiyao/p/13675101.html
Copyright © 2011-2022 走看看