目录
仿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创建
方式二:命令行的方式创建
python -m venv pyvenv38
PS:python命令此处不支持多版本共存的操作 python27 python36 python38,环境变量中谁靠前就用哪个解释器版本
激活:activate
关闭:deactivate(后面会详细解释)