纯手撸web框架

web框架的本质

浏览器 --- web框架 --- 数据库

理解1:web框架连接前端与数据库的中间介质

浏览器(客户端)---> web框架(服务端)

理解2:web框架是一个socket服务端

web框架就是一个服务端!

提供页面,从数据库中拿数据。

手写web框架

  1. 编写socket服务端代码

  2. 浏览器访问响应无效>>>:HTTP协议



    为什么显示响应无效?

    浏览器不认识服务端的响应 我们的服务端接受到浏览器的请求之后 返回一个字符串 浏览器不认识这种格式

    浏览器只支持http协议的数据格式(响应头 响应体)所以需要给字符串包装一下 加个响应头

    sock.send(b'HTTP/1.1 200 OK\r\n\r\n')

  3. 根据网址后缀的不同获取不同的页面内容

    输入网址:127.0.0.1:8080/login 跳转到登录页面

    输入网站:127.0.0.1:8080/rigister跳转到注册页面

  4. 想办法获取到用户输入的后缀>>>:请求数据

    要实现这个功能我们需要研究下浏览器发送的请求信息是什么?



    发现请求首行这个固定的位置,就有我们想要的信息(/login)。这个请求信息,经过解码就变成了一个字符串,我们可以使用正则或者split方法,将其匹配。

更多请求时的例子:



第一个是浏览器向127.0.0.1:8080/发送请求,第二个无需在意。

  1. 请求首行

    浏览器发送的请求也是有区别的:先介绍两种 GET、POST

    (请求首行:GET /login HTTP/1.1)

    5.1 GET请求

    朝别人索要数据 你把你的首页给我!向服务器索要html页面

    5.2 POST请求

    朝别人提交数据 注册登录时 将用户名密码交给服务端

  2. 处理请求数据获取网址后缀

    这时候我们就可以根据不同的请求数据 给浏览器发不同的消息了,写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!')

存在的问题

  1. socket代码过于重复
  2. 针对请求数据处理繁琐

    请求数据 我们只拿了一个数据 如果想拿更多的东西?那不是对请求 需要做更多的处理?
  3. 后缀匹配逻辑过于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文件上再返回给浏览器展示给用户看

  1. 将时间信息塞到html页面中?怎么塞??



    open方法r模式打开html 用变量data接受 data是个字符串!

    所以我们可以:

    读取html内容(字符串类型) 然后利用字符串替换(replace) 最后再返回给浏览器
  2. 先在html打个标记(类似占位符):

  3. 对标记做替换:

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基本操作的更多相关文章

  1. Django---Http协议简述和原理,HTTP请求码,HTTP请求格式和响应格式(重点),Django的安装与使用,Django项目的创建和运行(cmd和pycharm两种模式),Django的基础文件配置,Web框架的本质,服务器程序和应用程序(wsgiref服务端模块,jinja2模板渲染模块)的使用

    Django---Http协议简述和原理,HTTP请求码,HTTP请求格式和响应格式(重点),Django的安装与使用,Django项目的创建和运行(cmd和pycharm两种模式),Django的基 ...

  2. 【每天半小时学框架】——React.js的模板语法与组件概念

           [重点提前说:组件化与虚拟DOM是React.js的核心理念!]        先抛出一个论题:在React.js中,JSX语法提倡将 HTML 和 CSS 全都写入到JavaScrip ...

  3. Django(四)框架之第三篇模板语法

    https://www.cnblogs.com/yuanchenqi/articles/6083427.htm https://www.cnblogs.com/haiyan123/p/7725568. ...

  4. Django框架之第三篇模板语法(重要!!!)

    一.什么是模板? 只要是在html里面有模板语法就不是html文件了,这样的文件就叫做模板. 二.模板语法分类 一.模板语法之变量:语法为 {{ }}: 在 Django 模板中遍历复杂数据结构的关键 ...

  5. Django框架之第三篇模板语法

    一.什么是模板? 只要是在html里面有模板语法就不是html文件了,这样的文件就叫做模板. 二.模板语法分类 一.模板语法之变量:语法为 {{ }}: 在 Django 模板中遍历复杂数据结构的关键 ...

  6. 测开之路一百二十九:jinja2模板语法

    flask用的是jinja2模板,有自己特定的语法 形参: 在html里面留占位参数: {{ 参数名 }},后端传值时,参数名=参数值 <!DOCTYPE html><html la ...

  7. 【Vue】Vue框架常用知识点 Vue的模板语法、计算属性与侦听器、条件渲染、列表渲染、Class与Style绑定介绍与基本的用法

    Vue框架常用知识点 文章目录 Vue框架常用知识点 知识点解释 第一个vue应用 模板语法 计算属性与侦听器 条件渲染.列表渲染.Class与Style绑定 知识点解释 vue框架知识体系 [1]基 ...

  8. Flask框架(2)-JinJa2模板

    为了把业务逻辑和表现逻辑分开,Flask把表现逻辑移到JinJa2模板,模板是一个包含响应文本的文件.它用占位变量表示动态部分,其具体要从请求上下文才知道. 把真实值替换掉占位变量成为渲染,JinJa ...

  9. Django框架简介,wsgiref 与 jinja2 模块

    目录 框架简介 wsgiref模块 jinja2 模块 框架简介 Django是一个web开发框架,用来开发web应用,本质就是, web框架+socket服务端 MVC框架和MTV框架 MVC,全名 ...

  10. wsgiref模块、web框架、django框架简介

    """web框架:将前端.数据库整合到一起的基于互联网传输的python代码 web框架也可以简单的理解为是软件开发架构里面的'服务端'""" ...

随机推荐

  1. k8s更换网络插件:从flannel更换成calico

    卸载flannel 查看已安装的flannel的信息 # 查看CNI插件,可以得知使用的是flannel # cat /etc/cni/net.d/10-flannel.conflist { &quo ...

  2. 大数据技术之HBase原理与实战归纳分享-上

    @ 目录 概述 定义 特点 数据模型 概述 逻辑结构 物理存储结构 数据模型 应用场景 基础架构 安装 前置条件 部署 启动服务 高可用 Shell操作 基础操作 命令空间 DDL DML 概述 定义 ...

  3. JSP页面实现验证码校验

    目录 验证码校验分析 生成验证码 测试验证码 校验验证码 测试验证码校验 添加验证码刷新 在网页页面的使用中为防止"非人类"的大量操作和防止一些的信息冗余,增加验证码校验是许多网站 ...

  4. SDOI2017树点染色

    题目链接 发现1操作很像lct中的access,然后它每次染的又是一个新颜色,因此同一个颜色就在同一颗splay里了,且一个点到根的权值val[i]也就是到根路径上虚边的个数,然后看access时会对 ...

  5. C语言之走迷宫深度和广度优先(利用堆栈和队列)

    完成以下迷宫 利用二维数组储存每一个数组里的值,若是不能走则为1,若是可行就是0,走过了就设为2. 一般是再复制一个数组,用来记录. 堆栈的思想就是将一个点的上下左右都遍历一遍,若可行进栈,跳出遍历, ...

  6. c++算法竞赛常用板子集合(持续更新)

    前言 本文主要包含算法竞赛一些常用的板子,码风可能不是太好,还请见谅. 后续会继续补充没有的板子.当然我太菜了有些可能写不出来T^T 稍微有些分类但不多,原谅我QwQ 建议 Ctrl + F 以快速查 ...

  7. 3.pytest断言assert

    pytest使用的python自带的断言assert关键字,和unittest封装的assert断言不一样 原理:用来测试某个断言条件,如果断言条件为True,则程序将继续正常执行:但如果断言条件为假 ...

  8. Codeforces 1672 E. notepad.exe

    题意 这是一道交互题,有n个字符串,每个字符串长度:0-2000, n :0-2000 有一个机器对他进行排版,你可以给他一个每行的最大宽度w,那么每行只能放长度为w的字符: 每行相邻两个字符串之间至 ...

  9. ROS2时间同步(python)

    最近1周一直研究ROS2的时间同步,翻越很多博客,很少有人使用ROS2进行时间同步的代码,无奈不断尝试与源码阅读,终于将其搞定, 为此,本博客将介绍基于python的ROS2的时间同步方法. 本博客内 ...

  10. web前端学习之旅笔记01--HTML

    web前端学习之旅笔记01--HTML HTML最容易上手,但也易忘,实际开发中有时需要查阅官方文档,小伙伴们别忘了哟! HTML 教程 (w3school.com.cn) HTML是网页的骨架负责页 ...