web框架推导 wsgiref模块 jinja2模板语法 django框架简介 django基本操作
纯手撸web框架
web框架的本质
浏览器 --- web框架 --- 数据库
理解1:web框架连接前端与数据库的中间介质
浏览器(客户端)---> web框架(服务端)
理解2:web框架是一个socket服务端
web框架就是一个服务端!
提供页面,从数据库中拿数据。
手写web框架
编写socket服务端代码
浏览器访问响应无效>>>:HTTP协议
为什么显示响应无效?
浏览器不认识服务端的响应 我们的服务端接受到浏览器的请求之后 返回一个字符串 浏览器不认识这种格式
浏览器只支持http协议的数据格式(响应头 响应体)所以需要给字符串包装一下 加个响应头
sock.send(b'HTTP/1.1 200 OK\r\n\r\n')
根据网址后缀的不同获取不同的页面内容
输入网址:127.0.0.1:8080/login
跳转到登录页面
输入网站:127.0.0.1:8080/rigister
跳转到注册页面想办法获取到用户输入的后缀>>>:请求数据
要实现这个功能我们需要研究下浏览器发送的请求信息是什么?
发现请求首行这个固定的位置,就有我们想要的信息(/login)。这个请求信息,经过解码就变成了一个字符串,我们可以使用正则或者split方法,将其匹配。
更多请求时的例子:
第一个是浏览器向127.0.0.1:8080/
发送请求,第二个无需在意。
请求首行
浏览器发送的请求也是有区别的:先介绍两种 GET、POST
(请求首行:GET /login HTTP/1.1)
5.1 GET请求
朝别人索要数据 你把你的首页给我!向服务器索要html页面
5.2 POST请求
朝别人提交数据 注册登录时 将用户名密码交给服务端处理请求数据获取网址后缀
这时候我们就可以根据不同的请求数据 给浏览器发不同的消息了,写if分支结构:
我们上面发送的是字符串,也可以发送html文件:
rb模式读取html,以二进制格式(bytes)发送给浏览器。这样就完成了一个简单的web服务器。
代码如下:
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!')
存在的问题
- socket代码过于重复
- 针对请求数据处理繁琐
请求数据 我们只拿了一个数据 如果想拿更多的东西?那不是对请求 需要做更多的处理? - 后缀匹配逻辑过于LowB
后缀多的时候 :写100个if elif 后缀匹配逻辑太low
基于wsgiref模块
基本介绍
内置模块 很多web框架底层使用的模块
功能1:封装了socket代码
功能2:处理了请求、响应数据
(给字符串添加 响应头 给接收到的请求信息 自动处理成字典方便调用)
推导流程
1.固定代码启动服务端
看这行:
make_server(127.0.0.1, 8080, run)
一旦有浏览器向我们的服务器发送请求,自动触发run函数,自动给第三个参数run函数加括号调用并传参数,给run传的这个参数就是处理好的请求信息(大字典)
补充:我们make_server这里放的是run函数,能不能放一个对象?因为make_server可以自动加调用嘛。
这时候就要复习下了:
函数名加括号 ---> 函数执行
类名加括号 ---> 产生对象
对象加括号 ---> ???
对象加括号:理论上直接报错 但是如果你定义了双下__call__会自动触发!好就到这里,继续之前的思路。
还需要添加一行代码,我们的服务端才能起来:
server.serve_forever()
服务器等待请求中:
每次请求都是触发run函数,无论你用什么后缀:
2.查看处理之后的request大字典>>>:研究大字典键值对
里面path_info是我们要的网址后缀:
3.根据不同的网址后缀返回不同的内容
从request字典里面取值,写if逻辑判断:
针对/index页面返回字符串'index',在经过wsgiref模块自动加响应头,发送给浏览器。
4.立刻解决上述纯手撸的两个问题
socket代码过于重复 --> wsgirel模块帮你写
针对请求数据处理繁琐 --> wsgirel模块帮你打包成字典
5.针对最后一个问题代码如何优化
后缀匹配逻辑(if分支结构冗余)
代码:
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.网址后缀的匹配问题分析
问题1:if elif 随着页面增多而增多 有100个页面就要写100个if
问题2:每个后缀匹配成功后执行代码有多有少 万一有10000行代码怎么办,都放在一个if分支下面吗? 问题2很重要,慢慢拆分来解决!!
2.每个后缀匹配成功后执行的代码有多有少
我们现在相当于面条版,需要慢慢升级!
面条版 ---> 函数版 ---> 模块版
3.将分支的代码封装成一个个函数
将每个页面拆分成一个个函数:
4.将网址后缀与函数名做对应关系
url列表套元祖(解决if判断):
5.获取网址后缀循环匹配
一旦匹配成功应该结束for循环:
可能匹配结束之后,匹配不到,此时funcname=NONE
所以要加一个funcname的判断:
最终还是要看下面这个return:
6.如果想新增功能只需要先写函数再添加一个对应关系即可
res都是字符串!
这样写就相当于可以新增后缀 新增功能
核心是:获取网址后缀for循环匹配
还不够完美!这些功能函数全写在一起了!要根据功能不同进行拆分!
7.根据不同的功能拆分成不同的py文件
views.py 存储核心业务逻辑(功能函数)
urls.py 存储网址后缀与函数名对应关系
templates目录 存储html页面文件
run函数导入urls.py:
urls.py带入views:
请求:RUN函数 ---> url选择器 ---> views
响应:RUN return <----views return
8.为了使函数体代码中业务逻辑有更多的数据可用
将request大字典转手传给这个函数(可用不用但是不能没有)
ps: funcname的res返回值是包含html信息的字符串 经过编码就可以发送给浏览器
总结
核心思路就是解决三个问题:
1socket代码总是要写!
2.请求数据拿不全 我们只拿了一个数据 如果想拿更多的东西?那不是对请求 需要做更多的处理?
3.后缀多的时候 :写100个if elif 后缀匹配逻辑太low
动静态网页
动态网页
页面数据来源于后端
静态网页
页面数据直接写死
简单实现动态网页:
访问某个网址后缀 后端代码获取当前时间 并将该时间传到html文件上再返回给浏览器展示给用户看
- 将时间信息塞到html页面中?怎么塞??
open方法r模式打开html 用变量data接受 data是个字符串!
所以我们可以:
读取html内容(字符串类型) 然后利用字符串替换(replace) 最后再返回给浏览器 - 先在html打个标记(类似占位符):
- 对标记做替换:
jinja2模块实现动态网页:
需求:将字典传递给页面内容 并且在页面上还可以通过类似于后端的操作方式操作该数据(就是在前端可以写python语法 使用for循环 if判断等)
jinja2模块
这是一个第三方模块。
他可以让你在html上面写python代码。
原理:所谓的模板语法 还是只有后端才认识 浏览器不认识(浏览器只支持html\css\js) jinja2模块可以自动解析html页面的模板语法 添加数据 生成新的html页面 然后再交给前端执行
导入:
写模板语法:
render函数的第一个参数是在模板语法中使用的变量名,第二个参数是这个变量名对应的数据值。
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>
前端、后端、数据库三者联动
简单的说,就是在功能函数中使用pymysql模块。
然后将得到的数据,通过模板语法,动态添加到html页面上。模板语法对html页面做一些操作,产生新的html页面,这个过程称为‘渲染’。
推导流程
添加url选择器 添加函数 写函数 添加pymysql:
我们还要将数据表的样式搞下 用cdn 暂时别用本地,现在还不会用,可能会出问题:
将数据库的数据传给模板(Template类产生的对象):
编写html,以及模板语法:
使用for循环,书写表单体:
总结
python主流web框架
"""
作为小白的你 初学阶段不要混着学 很容易走火入魔
"""
1.django
大而全 自身自带的功能组件非常的多 类似于航空母舰
2.flask
小而精 自身自带的功能组件非常的少 类似于游骑兵
几乎所有的功能都需要依赖于第三方模块
3.tornado
异步非阻塞 速度极快效率极高甚至可以充当游戏服务端
封装多进程多线程 可选同步异步
ps:sanic、fastapi...
django简介
版本问题
django1.X:同步 1.11
django2.X:同步 2.2
django3.X:支持异步 3.2
django4.X:支持异步 4.2
ps:版本之间的差异其实不大 主要是添加了额外的功能
运行django注意事项
1.django项目中所有的文件名目录名不要出现中文
2.计算机名称尽量也不要出现中文
3.一个pycharm尽量就是一个完整的项目(不要嵌套 不要叠加)
4.不同版本的python解释器与不同版本的django可能会出现小问题
计算机名不能为中文:会报编码错误
django基本使用
1. 下载
pip3 install django 默认最新版
pip3 install django==版本号 指定版本
pip3 install django==2.2.22
pip下载模块会自动解决依赖问题(会把关联需要用到的模块一起下了)
2. 验证
django-admin
django下好了 会自动在python/scrpit目录下添加django-admin.exe
如果输入django-admin终端显示找不到文件
先检查环境变量是否添加scrpit目录 再去该目录下查看exe文件是否存在
如果exe不存在,则考虑重装django。
3. 常见命令
1.创建django项目
django-admin startproject 项目名
2.启动django项目
cd 项目名
python38 manage.py runserver ip:port
manage文件相当于上面写的run函数那个文件。
默认会在8000段口起了一个服务
4. pycharm自动创建django项目
会自动创建templates文件夹 但是配置文件中可能会报错
os.path.join(BASE_DIR,'templates')
如下是pycharm创建的settings:
修改一下(路径分隔符错误):
注意应该改成这样:[os.path.join(BASE_DIR,'templates'),]
用pycharm提供的项目启动:
可以改端口:
不小心把启动文件删掉了:
自己添加django server
django app的概念
django类似于是一所大学 app类似于大学里面的各个学院
django里面的app类似于某个具体的功能模块
user app 所有用户相关的都写在user app下
goods app 所有商品相关的都写在goods app下
命令行创建应用
python38 manage.py startapp 应用名
pycharm创建应用
新建django项目可以默认创建一个 并且自动注册
"""
创建的app一定要去settings.py中注册
INSTALLED_APPS = [
'app01.apps.App01Config',
'app02'
]
"""
pycharm自动创建app
pycharm创建django项目的时候自动帮你创一个app:
好处:app需要在settings注册 (给校长打招呼)
pycharm创建的应用会自动注册一下
创建的app一定要去settings注册,不然django不带你玩。
注册时有两种写法:完整和简写
pycharm提供一个manage.py终端:
django主要目录结构
django项目目录名>>>
django项目同名目录>>>
settings.py 配置文件
urls.py 存储网址后缀与函数名对应关系(不严谨)
wsgi.py wsgiref网关文件
db.sqlite3文件 django自带的小型数据库(项目启动之后才会出现)
manage.py 入口文件(命令提供)
app应用目录>>>
migrations目录 存储数据库相关记录
admin.py django内置的admin后台管理功能
apps.py 注册app相关
models.py 与数据库打交道的(非常重要)
tests.py 测试文件
views.py 存储功能函数(不严谨)
templates目录>>>
存储html文件(命令行不会自动创建 pycharm会)
配置文件settings中还需要配置路径
[os.path.join(BASE_DIR,'templates'),]
"""
重要名词讲解
网址后缀 路由
函数 视图函数
类 视图类
urls.py 路由层
views.py 视图层
models.py 模型层
templates 模板层
"""
小江狗(django)必会三板斧(需要补充)
from django.shortcuts import render,HttpResponse,redirect
HttpResponse 返回字符串类型的数据
render 返回html页面并且支持传值
redirect 重定向
web框架推导 wsgiref模块 jinja2模板语法 django框架简介 django基本操作的更多相关文章
- Django---Http协议简述和原理,HTTP请求码,HTTP请求格式和响应格式(重点),Django的安装与使用,Django项目的创建和运行(cmd和pycharm两种模式),Django的基础文件配置,Web框架的本质,服务器程序和应用程序(wsgiref服务端模块,jinja2模板渲染模块)的使用
Django---Http协议简述和原理,HTTP请求码,HTTP请求格式和响应格式(重点),Django的安装与使用,Django项目的创建和运行(cmd和pycharm两种模式),Django的基 ...
- 【每天半小时学框架】——React.js的模板语法与组件概念
[重点提前说:组件化与虚拟DOM是React.js的核心理念!] 先抛出一个论题:在React.js中,JSX语法提倡将 HTML 和 CSS 全都写入到JavaScrip ...
- Django(四)框架之第三篇模板语法
https://www.cnblogs.com/yuanchenqi/articles/6083427.htm https://www.cnblogs.com/haiyan123/p/7725568. ...
- Django框架之第三篇模板语法(重要!!!)
一.什么是模板? 只要是在html里面有模板语法就不是html文件了,这样的文件就叫做模板. 二.模板语法分类 一.模板语法之变量:语法为 {{ }}: 在 Django 模板中遍历复杂数据结构的关键 ...
- Django框架之第三篇模板语法
一.什么是模板? 只要是在html里面有模板语法就不是html文件了,这样的文件就叫做模板. 二.模板语法分类 一.模板语法之变量:语法为 {{ }}: 在 Django 模板中遍历复杂数据结构的关键 ...
- 测开之路一百二十九:jinja2模板语法
flask用的是jinja2模板,有自己特定的语法 形参: 在html里面留占位参数: {{ 参数名 }},后端传值时,参数名=参数值 <!DOCTYPE html><html la ...
- 【Vue】Vue框架常用知识点 Vue的模板语法、计算属性与侦听器、条件渲染、列表渲染、Class与Style绑定介绍与基本的用法
Vue框架常用知识点 文章目录 Vue框架常用知识点 知识点解释 第一个vue应用 模板语法 计算属性与侦听器 条件渲染.列表渲染.Class与Style绑定 知识点解释 vue框架知识体系 [1]基 ...
- Flask框架(2)-JinJa2模板
为了把业务逻辑和表现逻辑分开,Flask把表现逻辑移到JinJa2模板,模板是一个包含响应文本的文件.它用占位变量表示动态部分,其具体要从请求上下文才知道. 把真实值替换掉占位变量成为渲染,JinJa ...
- Django框架简介,wsgiref 与 jinja2 模块
目录 框架简介 wsgiref模块 jinja2 模块 框架简介 Django是一个web开发框架,用来开发web应用,本质就是, web框架+socket服务端 MVC框架和MTV框架 MVC,全名 ...
- wsgiref模块、web框架、django框架简介
"""web框架:将前端.数据库整合到一起的基于互联网传输的python代码 web框架也可以简单的理解为是软件开发架构里面的'服务端'""" ...
随机推荐
- k8s更换网络插件:从flannel更换成calico
卸载flannel 查看已安装的flannel的信息 # 查看CNI插件,可以得知使用的是flannel # cat /etc/cni/net.d/10-flannel.conflist { &quo ...
- 大数据技术之HBase原理与实战归纳分享-上
@ 目录 概述 定义 特点 数据模型 概述 逻辑结构 物理存储结构 数据模型 应用场景 基础架构 安装 前置条件 部署 启动服务 高可用 Shell操作 基础操作 命令空间 DDL DML 概述 定义 ...
- JSP页面实现验证码校验
目录 验证码校验分析 生成验证码 测试验证码 校验验证码 测试验证码校验 添加验证码刷新 在网页页面的使用中为防止"非人类"的大量操作和防止一些的信息冗余,增加验证码校验是许多网站 ...
- SDOI2017树点染色
题目链接 发现1操作很像lct中的access,然后它每次染的又是一个新颜色,因此同一个颜色就在同一颗splay里了,且一个点到根的权值val[i]也就是到根路径上虚边的个数,然后看access时会对 ...
- C语言之走迷宫深度和广度优先(利用堆栈和队列)
完成以下迷宫 利用二维数组储存每一个数组里的值,若是不能走则为1,若是可行就是0,走过了就设为2. 一般是再复制一个数组,用来记录. 堆栈的思想就是将一个点的上下左右都遍历一遍,若可行进栈,跳出遍历, ...
- c++算法竞赛常用板子集合(持续更新)
前言 本文主要包含算法竞赛一些常用的板子,码风可能不是太好,还请见谅. 后续会继续补充没有的板子.当然我太菜了有些可能写不出来T^T 稍微有些分类但不多,原谅我QwQ 建议 Ctrl + F 以快速查 ...
- 3.pytest断言assert
pytest使用的python自带的断言assert关键字,和unittest封装的assert断言不一样 原理:用来测试某个断言条件,如果断言条件为True,则程序将继续正常执行:但如果断言条件为假 ...
- Codeforces 1672 E. notepad.exe
题意 这是一道交互题,有n个字符串,每个字符串长度:0-2000, n :0-2000 有一个机器对他进行排版,你可以给他一个每行的最大宽度w,那么每行只能放长度为w的字符: 每行相邻两个字符串之间至 ...
- ROS2时间同步(python)
最近1周一直研究ROS2的时间同步,翻越很多博客,很少有人使用ROS2进行时间同步的代码,无奈不断尝试与源码阅读,终于将其搞定, 为此,本博客将介绍基于python的ROS2的时间同步方法. 本博客内 ...
- web前端学习之旅笔记01--HTML
web前端学习之旅笔记01--HTML HTML最容易上手,但也易忘,实际开发中有时需要查阅官方文档,小伙伴们别忘了哟! HTML 教程 (w3school.com.cn) HTML是网页的骨架负责页 ...