mini-web框架的实现

一、面向对象的多进程web服务器

"""
webserver.py
使用面向对象的方法来做多进程的web server
"""

import socket
import re
import multiprocessing


class WSGIServer(object):
    """服务器对象"""
    def __init__(self):
        # 创建套接字
        self.tcp_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        self.tcp_server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        # 绑定ip和port
        self.tcp_server.bind(("", 8080))
        # 设置成监听模式
        self.tcp_server.listen(128)

    def service_client(self, new_socket):
        # 接受数据
        data = new_socket.recv(1024).decode("utf-8")
        # print(data.decode("utf-8"))
        data_list = data.splitlines()
        # print(data_list[0])

        # 正则匹配出文件名   'GET /img/home/tour8.jpg HTTP/1.1',
        ret = re.match(r"[^/]+(/[^ ]*)",data_list[0])
        print(ret)
        file_name = ""
        if ret:
            file_name = ret.group(1)
            print(file_name)
            if file_name == "/":
                file_name = "/index.html"
        try:
            # 打开文件
            f = open("..//..//static/html/Trave" + file_name, "rb")
        except:

            response = "HTTP/1.1 404 NOT FOUND\r\n"
            response += "\r\n"
            response += "Not Found"
            # 发送文件 header
            new_socket.send(response.encode("utf-8"))
        else:
            # 传递数据
            response = "HTTP:/1.1 200 OK\r\n"
            response += "\r\n"
            data_contend = f.read()
            f.close()
            # 发送文件 header
            new_socket.send(response.encode("utf-8"))
            # 发送文件body
            new_socket.send(data_contend)
        # 关闭套接字
        new_socket.close()

    def run_forerver(self):
        """使程序一直运行"""
        while True:
            # 等待客户端连接
            new_client, addr = self.tcp_server.accept()
            p = multiprocessing.Process(target=self.service_client, args=(new_client, ))
            # 接受客户端的消息  回复客户端的消息
            # service_client(new_client)
            p.start()
            new_client.close()  # 多进程中会复制一份这个变量,所以要在主进程中也要关闭
        self.tcp_server.close()


def main():
    """控制整体流程 --> 创建一个对象 --> 调用这个对象的run_forerver方法"""
    wsgi_server = WSGIServer()
    wsgi_server.run_forerver()


if __name__ == "__main__":
    main()

二、让web服务器支持动态解析

  1. 解释:当浏览器请求文件以.py结尾的就当做动态网页

  2. 代码

    """
    webserver.py
    使用面向对象的方法来做多进程的web server 继承了动态的解析功能 
    当请求文件结尾以.py结尾的为动态网页
    
    """
    
    import socket
    import re
    import multiprocessing
    import time
    
    
    class WSGIServer(object):
        """服务器对象"""
        def __init__(self):
            # 创建套接字
            self.tcp_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
            self.tcp_server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
            # 绑定ip和port
            self.tcp_server.bind(("", 8080))
            # 设置成监听模式
            self.tcp_server.listen(128)
    
        def service_client(self, new_socket):
            # 接受数据
            data = new_socket.recv(1024).decode("utf-8")
            # print(data.decode("utf-8"))
            data_list = data.splitlines()
            # print(data_list[0])
    
            # 正则匹配出文件名   'GET /img/home/tour8.jpg HTTP/1.1',
            ret = re.match(r"[^/]+(/[^ ]*)",data_list[0])
            print(ret)
            file_name = ""
            if ret:
                file_name = ret.group(1)
                print(file_name)
                if file_name == "/":
                    file_name = "/index.html"
    
            if not file_name.endswith('.py'):
                try:
                    # 打开文件
                    f = open("..//..//static/html/Trave" + file_name, "rb")
                except:
    
                    response = "HTTP/1.1 404 NOT FOUND\r\n"
                    response += "\r\n"
                    response += "Not Found"
                    # 发送文件 header
                    new_socket.send(response.encode("utf-8"))
                else:
                    # 传递数据
                    response = "HTTP:/1.1 200 OK\r\n"
                    response += "\r\n"
                    data_contend = f.read()
                    f.close()
                    # 发送文件 header
                    new_socket.send(response.encode("utf-8"))
                    # 发送文件body
                    new_socket.send(data_contend)
            else:
                # 以.py结尾的则为动态的获取的页面
                header = "HTTP:/1.1 200 OK\r\n"
                header += "\r\n"
                
                body = "nihao : %s" % time.ctime()
    
                response = header + body
                # 发送消息
                new_socket.send(response.encode("utf-8"))
            # 关闭套接字
            new_socket.close()
    
        def run_forerver(self):
            """使程序一直运行"""
            while True:
                # 等待客户端连接
                new_client, addr = self.tcp_server.accept()
                p = multiprocessing.Process(target=self.service_client, args=(new_client, ))
                # 接受客户端的消息  回复客户端的消息
                # service_client(new_client)
                p.start()
                new_client.close()  # 多进程中会复制一份这个变量,所以要在主进程中也要关闭
            self.tcp_server.close()
    
    
    def main():
        """控制整体流程 --> 创建一个对象 --> 调用这个对象的run_forerver方法"""
        wsgi_server = WSGIServer()
        wsgi_server.run_forerver()
    
    
    if __name__ == "__main__":
        main()
    

三、将服务器与逻辑处理分开实现解耦

  1. 版本一:
    1. webserver.py 服务器代码

      """
      
      使用面向对象的方法来做多进程的web server 继承了动态的解析功能 
      当请求文件结尾以.py结尾的为动态网页
      
      将web框架与逻辑处理的代码分开
      
      """
      
      import socket
      import re
      import multiprocessing
      import time
      import mini_frame
      
      
      class WSGIServer(object):
          """服务器对象"""
          def __init__(self):
              # 创建套接字
              self.tcp_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
              self.tcp_server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
              # 绑定ip和port
              self.tcp_server.bind(("", 8080))
              # 设置成监听模式
              self.tcp_server.listen(128)
      
          def service_client(self, new_socket):
              # 接受数据
              data = new_socket.recv(1024).decode("utf-8")
              # print(data.decode("utf-8"))
              data_list = data.splitlines()
              # print(data_list[0])
      
              # 正则匹配出文件名   'GET /img/home/tour8.jpg HTTP/1.1',
              ret = re.match(r"[^/]+(/[^ ]*)",data_list[0])
              print(ret)
              file_name = ""
              if ret:
                  file_name = ret.group(1)
                  
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,让我来为您介绍一个简单的 Python Web 框架,它类似于 Django,名字叫做 MiniWebMiniWeb 实现了一个基本的 Web 框架功能,包括 URL 路由、请求处理、模板渲染等。下面是 MiniWeb 的代码示例: ```python # 引入必要的库 import re import os import sys import time import socket import select # 定义 MiniWeb 类 class MiniWeb: # 初始化方法 def __init__(self, host='', port=80): self.host = host self.port = port self.routes = [] # 添加路由方法 def route(self, pattern): def decorator(handler): self.routes.append((re.compile(pattern), handler)) return handler return decorator # 处理请求方法 def handle_request(self, request): # 提取请求方法和路径 method, path, _ = request.split(' ', 2) # 遍历路由规则 for pattern, handler in self.routes: # 如果路径匹配路由规则,则调用对应的处理函数 if pattern.match(path): return handler(request) # 如果没有匹配的路由规则,则返回 404 错误 return 'HTTP/1.1 404 Not Found\r\n\r\n' # 运行框架方法 def run(self): # 创建套接字并绑定地址 server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) server_socket.bind((self.host, self.port)) server_socket.listen(5) # 循环等待客户端连接 while True: # 使用 select 函数监听套接字 readable, _, _ = select.select([server_socket], [], [], 1) if readable: # 接受客户端连接并处理请求 client_socket, client_address = server_socket.accept() request = client_socket.recv(1024).decode() response = self.handle_request(request) client_socket.sendall(response.encode()) client_socket.close() # 定义路由处理函数 web = MiniWeb() @web.route('^/$') def index(request): return 'HTTP/1.1 200 OK\r\n\r\nHello, World!' @web.route('^/hello/(.*)$') def hello(request): name = request.split('/')[2] return f'HTTP/1.1 200 OK\r\n\r\nHello, {name}!' # 运行框架 if __name__ == '__main__': web.run() ``` 以上是 MiniWeb 的代码示例,您可以通过调用 `route` 方法来添加路由规则,然后编写对应的路由处理函数。当客户端发起请求时,MiniWeb 会根据路由规则来调用对应的路由处理函数,并返回处理结果。 当然,MiniWeb 还需要进一步的完善和优化,例如添加静态文件处理、错误处理、中间件等功能。但它足以作为一个简单的 Python Web 框架示例,帮助您更好地理解 Web 框架实现原理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值