目录
仿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 socketserver = socket.socket() # TCP UDPserver.bind(("127.0.0.1", 8080)) # IP PORTserver.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_serverdef 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 Templatedef 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{{ d1 }}
{{ d1.name }}
{{ d1["age"] }}
{{ d1.get("person_list") }}
五、动静态网页
动态网页:页面数据来源于后端
静态网页:页面数据直接写死的,就是我们常见的: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创建
方式二:命令行的方式创建
python -m venv pyvenv38
PS:python命令此处不支持多版本共存的操作 python27 python36 python38,环境变量中谁靠前就用哪个解释器版本激活:activate关闭:deactivate(后面会详细解释)
关键词: