仿Django框架-基于wsgiref模块和jinja2模块写一个简单的框架 主流框架简介 动静态网页 Python虚拟环境

发布时间 2023-04-25 16:16:24作者: 南柯unique

目录

仿Django框架-基于wsgiref模块和jinja2模块写一个简单的框架

一、前期需要的了解背景知识

  • web框架的本质
    • 理解1:连接前端与数据库的中间介质
    • 理解2:socket服务端
  • 手写web框架的大概思路
    1.编写socket服务端代码
    2.浏览器访问响应无效>>>:HTTP协议
    3.根据网址后缀的不同获取不同的页面内容
    4.想办法获取到用户输入的后缀>>>:请求数据
    5.请求首行>>>GET /login HTTP/1.1
  •  GET请求   朝别人索要数据
    
  •  POST请求  朝别人提交数据
    

    6.处理请求数据获取网址后缀
    问题:
    1.socket代码过于重复
    2.针对请求数据处理繁琐
    3.后缀匹配逻辑过于LowB

基于socket模块写的服务端

import socket


server = socket.socket()  # TCP UDP
server.bind(('127.0.0.1', 8080))  # IP PORT
server.listen(5)  # 半连接池


while True:
    sock, address = server.accept()  # 等待连接
    data = sock.recv(1024)  # 字节(bytes)
    # print(data.decode('utf8'))  # 解码打印
    sock.send(b'HTTP/1.1 200 OK\r\n\r\n')
    data_str = data.decode('utf8')  # 先转换成字符串
    target_url = data_str.split(' ')[1]  # 按照空格切割字符串并取索引1对应的数据
    # print(target_url)  # /index /login /reg
    if target_url == '/index':
        # sock.send(b'index page')
        with open(r'myhtml01.html','rb') as f:
            sock.send(f.read())
    elif target_url == '/login':
        sock.send(b'login page')
    else:
        sock.send(b'home page!')

二、基于wsgiref模块做优化

  • wsgiref功能简单介绍
    内置模块 很多web框架底层使用的模块

    • 功能1:封装了socket代码
    • 功能2:处理了请求数据
      1.固定代码启动服务端
      2.查看处理之后的request大字典
      3.根据不同的网址后缀返回不同的内容>>>:研究大字典键值对
      4.立刻解决上述纯手撸的两个问题
      5.针对最后一个问题代码如何优化
  • 代码编写

from wsgiref.simple_server import make_server


def run(request, response):
    """
    :param request: 请求相关数据
    :param response: 响应相关数据
    :return: 返回给客户端的真实数据
    """
    response('200 OK', [])  # 固定格式 不用管它
    # print(request)  是一个处理之后的大字典
    path_info = request.get('PATH_INFO')
    if path_info == '/index':
        return [b'index']
    elif path_info == '/login':
        return [b'login']
    return [b'hello wsgiref module']


if __name__ == '__main__':
    server = make_server('127.0.0.1', 8080, run)  # 实时监听127.0.0.1:8080 一旦有请求过来自动给第三个参数加括号并传参数调用
    server.serve_forever()  # 启动服务端

三、代码封装优化

1.网址后缀的匹配问题
2.每个后缀匹配成功后执行的代码有多有少,所以可以分成面条版 函数版 模块版
3.将分支的代码封装成一个个函数
4.将网址后缀与函数名做对应关系
5.获取网址后缀循环匹配
6.如果想新增功能只需要先写函数再添加一个对应关系即可
7.根据不同的功能拆分成不同的py文件

  •   views.py		存储核心业务逻辑(功能函数)
    
  •   urls.py			存储网址后缀与函数名对应关系
    
  •   templates目录	   存储html页面文件
    

8.为了使函数体代码中业务逻辑有更多的数据可用,将request大字典转手传给这个函数(可用不用但是不能没有)

四、jinja2模块

  • 命令行下载
    pip3 install jinja2

  • 使用

from jinja2 import Template


def get_dict_func(request):
    user_dict = {'name': 'jason', 'age': 18, 'person_list': ['阿珍', '阿强', '阿香', '阿红']}
    with open(r'templates/get_dict_page.html', 'r', encoding='utf8') as f:
        data = f.read()
    temp_obj = Template(data)  # 将页面数据交给模板处理
    res = temp_obj.render({'d1': user_dict})  # 给页面传了一个 变量名是d1值是字典数据的数据
    return res

<p>{{ d1 }}</p>
<p>{{ d1.name }}</p>
<p>{{ d1['age'] }}</p>
<p>{{ d1.get('person_list') }}</p>

五、动静态网页

  •   动态网页:页面数据来源于后端
    
  •   静态网页:页面数据直接写死的,就是我们常见的:CSS文件、JS文件、img文件、我们的第三方前端框架:bootstrap相关的
    
  • 了解概念
    1.访问某个网址后缀 后端代码获取当前时间 并将该时间传到html文件上再返回给浏览器展示给用户看
    读取html内容(字符串类型) 然后利用字符串替换 最后再返回给浏览器
    2.将字典传递给页面内容 并且在页面上还可以通过类似于后端的操作方式操作该数据
    模板语法>>>:jinja2模块

六、Python主流web框架

  • django

	大而全 自身自带的功能组件非常的多 类似于航空母舰 		
  • flask

	小而精 自身自带的功能组件非常的少 类似于游骑兵,几乎所有的功能都需要依赖于第三方模块 
  • tornado

	异步非阻塞 速度极快效率极高甚至可以充当游戏服务端

ps:还有sanic、fastapi...
提示:作为小白的你 初学阶段不要混着学 很容易走火入魔

七、虚拟环境

  • 为什么要使用虚拟环境?
    项目1需要使用:django1.11 python38
    项目2需要使用:django2.22 pymysql requests python38
    项目3需要使用:django3.22 request_html flask urllib3 python38

    实际开发项目中我们只会给项目配备所需的环境,不需要的一概不配!!!
    虚拟环境:能够针对相同版本的解释器创建多个分身 每个分身可以有自己独立的环境

  • 创建虚拟环境:(每创建一个虚拟环境就相当于重新下载了一个全新的解释器)
    方式一:pycharm创建
    image

方式二:命令行的方式创建


	python -m venv pyvenv38

PS:python命令此处不支持多版本共存的操作 python27 python36 python38,环境变量中谁靠前就用哪个解释器版本
激活:activate
关闭:deactivate(后面会详细解释)