• 静态服务器


    2. [难点]HTTP协议的介绍

    • 全称:超文本传输协议

    • 作用:规定了浏览器和web服务器数据传输的格式

    3. [重点]URL

    4. [重点]查看HTTP协议的通信过程

    • 谷歌浏览器开发者工具的使用

      • F12

      • 右键+检查+network

    • HTTP协议的通信过程

      • 一次请求报文

      • 一次响应报文

    5. [难点]HTTP请求报文

    • 注意:除了请求体外的每行结尾都有

    • HTTP请求报文:

      • HTTP GET请求报文分析:

        • 请求行

          • GET /xxx/xxx.html HTTP/1.1

          • 请求方法: GET

          • 请求资源路径:/xxx/xxx.html

          • 协议版本:HTTP/1.1

        • 请求头

          • 键:值

        • 空行

      • HTTP POST请求报文分析:

        • 请求行

          • POST /xxx/xx.html HTTP/1.1

        • 请求头

          • 键:值

        • 空行

        • 请求体

          • username=abc&password=xxx...

    6. [难点]HTTP响应报文

    • 注意:除了响应体外,每一行结尾都是

    • HTTP 响应报文:

      • 响应行

        • HTTP/1.1 200 OK

        • 协议版本: HTTP/1.1

        • 状态码: 20

        • 状态描述: OK

      • 响应头

        • 键:值

        • Server: BWS/1.1

      • 空行

      • 响应体

        • 页面代码....

    • HTTP 状态码:

      • 200

      • 307

      • 400

      • 404

      • 500

     

     

    2. 静态Web服务器-命令行启动动态绑定端口号的示例代码

    import socket
    import threading
    import sys
    
    
    # 定义web服务器类
    class HttpWebServer(object):
        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
    
        # 处理客户端的请求
        @staticmethod
        def handle_client_request(new_socket):
            # 代码执行到此,说明连接建立成功
            recv_client_data = new_socket.recv(4096)
            if len(recv_client_data) == 0:
                print("关闭浏览器了")
                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:
                # 动态打开指定文件
                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: PWS1.0
    "
                with open("static/error.html", "rb") as file:
                    file_data = file.read()
                # 响应体
                response_body = file_data
    
                # 拼接响应报文
                response_data = (response_line + response_header + "
    ").encode("utf-8") + response_body
                # 发送数据
                new_socket.send(response_data)
            else:
                # 响应行
                response_line = "HTTP/1.1 200 OK
    "
                # 响应头
                response_header = "Server: PWS1.0
    "
    
                # 响应体
                response_body = file_data
    
                # 拼接响应报文
                response_data = (response_line + response_header + "
    ").encode("utf-8") + response_body
                # 发送数据
                new_socket.send(response_data)
            finally:
                # 关闭服务与客户端的套接字
                new_socket.close()
    
        # 启动web服务器进行工作
        def start(self):
            while True:
                # 等待接受客户端的连接请求
                new_socket, ip_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("执行命令如下: python3 xxx.py 8000")
            return
    
        # 判断字符串是否都是数字组成
        if not sys.argv[1].isdigit():
            print("执行命令如下: python3 xxx.py 8000")
            return
    
        # 获取终端命令行参数
        port = int(sys.argv[1])
        # 创建web服务器对象
        web_server = HttpWebServer(port)
        # 启动web服务器进行工作
        web_server.start()
    
    
    if __name__ == '__main__':
        main()
  • 相关阅读:
    oracle,sql server count函数 存储过程 判断 行数 注意事项
    js 跨域访问 获取验证码图片 获取header 自定义属性
    开发作中常用,实用工具推荐!
    phpcms
    php基础
    jQuery , js 写选项卡
    js, jquery实现全选,反选
    jQuery选择器
    学习jQuery
    javascript 与 java继承问题
  • 原文地址:https://www.cnblogs.com/zhangwei112/p/13586442.html
Copyright © 2020-2023  润新知