zoukankan      html  css  js  c++  java
  • 【Python高级编程040 ● 静态web服务器 ● 静态Web服务器-命令行启动动态绑定端口号】


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

    Author : AI菌


    【内容讲解】

    一、需求
        获取终端命令行参数,获取端口号,然后在web服务器程序动态绑定端口号
    
    二、实现步骤:
        1、获取执行python程序的终端命令行参数
        2、判断参数的类型,设置端口号必须是整型
        3、给Web服务器类的初始化方法添加一个端口号参数,用于绑定端口号

    【代码演示】

    """
    一、需求
        获取终端命令行参数,获取端口号,然后在web服务器程序动态绑定端口号
    
    二、实现步骤:
        1、获取执行python程序的终端命令行参数
        2、判断参数的类型,设置端口号必须是整型
        3、给Web服务器类的初始化方法添加一个端口号参数,用于绑定端口号
    """
    
    import socket
    import threading
    import sys
    
    
    # 1、定义web服务器类
    class HttpWebServer(object):
        # 2、提供Web服务器的初始化方法__init__,在初始化方法里面创建socket对象
        def __init__(self, port):
            # 创建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(("", port))
            # 设置监听
            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():
        # 控制台打印获取到的参数
        print(sys.argv)
        # 判断命令行参数是否等于2,
        if len(sys.argv) != 2:
            print("按照如下格式输入命令: python xxx.py 8000")
            return
        # 判断字符串是否都是数字组成
        if not sys.argv[1].isdigit():
            print("按照如下格式输入命令: python xxx.py 8000")
            return
        # 获取终端命令行参数
        port = int(sys.argv[1])
    
        # 创建web服务器对象
        web_server = HttpWebServer(port)
        # 启动web服务器进行工作
        web_server.start()
    
    
    if __name__ == '__main__':
        main()

    【运行结果】

     在终端使用命令行启动程序(注意:不是在IDE中启动):

     

    在浏览器,使用命令行参数指定的端口,正常访问web服务器:

    【往期精彩】

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

    【加群交流】



  • 相关阅读:
    火狐下,td 的 bug;
    在centos6.3用yum安装redis
    Linux系统下修改环境变量PATH路径的三种方法
    网络文件常常提到类似"./run.sh"的数据,这个命令的意义是什么?
    DataGridView的自定义列排序
    【Linux】linux常用基本命令
    用户名 不在 sudoers文件中,此事将被报告。(转)
    将XML文件中的内容转换为Json对象
    C# 加载xml文档文件及加载xml字符串
    增加不存在的列
  • 原文地址:https://www.cnblogs.com/hezhiyao/p/13675275.html
Copyright © 2011-2022 走看看