请求生命周期流程图

django请求生命周期流程图

路由层之路由匹配

我们都知道,路由层是匹配对应关系用的,那么它是怎么匹配上的呢?

urlpatterns = [
url(r'^index/', views.index),
]

url方法的第一个参数其实是一个正则表达式,只要用户输入的地址后缀与内容匹配上,就会执行对应的视图函数。

并且django有一个二次追加斜杠机制,如果你在输入地址后缀时末尾没有加斜杠,django会先匹配一次,如果匹配不上,那么django还会让浏览器默认加斜杠再次发送请求,比如:

# 第一次匹配
http://127.0.0.1:8000/index # 匹配不上加斜杠再次发送请求
http://127.0.0.1:8000/index/

如果想要取消这个斜杠机制,那么可以在settings.py加上:

APPEND_SLASH = False

因为是正则表达式,所以比如像上述代码中的例子,我在地址栏输入:

http://127.0.0.1:8000/index/abcde/66

或者:

http://127.0.0.1:8000/index/

都是可以匹配到正则:^index/ 的

拓展

让用户不需要输入地址后缀就可以访问到指定页面:

urlpatterns = [
url(r'^$', views.index),
]

也可以定义一个尾页, 用户输入一个没有对应关系的直接返回:

urlpatterns = [
url(r'.*',views.error),
]

无名有名分组

无名分组

我们都知道,路由匹配成功之后就会调用视图函数,并且默认情况下会自动给视图函数传递一个request位置参数。

但是如果正则表达式中加了括号,那么就不止传递一个request位置参数了,还会传递括号内正则表达式匹配到的内容。

urlpatterns = [
url(r'^index/(.*)', views.index),
]

如果正则加了一个括号,那么视图函数就需要多写一个形参用于接收匹配到的内容,不然会报错。

def index(request, aa):
return HttpResponse(aa)

这个就是无名分组。

有名分组

python中的正则表达式加括号是可以给括号内的内容进行命名的,这里的正则表达式自然也可以。

给括号内的内容命名为address:

urlpatterns = [
url(r'^index/(?P<address>.*)', views.index),
]

给括号内命名后,视图函数就需要一个名字相同的形参接收:

def index(request, address):
return HttpResponse(address)

这个就是有名分组。

补充说明

1.无名分组需要一个形参接收括号内的内容,名字随意。

2.有名分组也需要一个形参接收括号内的内容,名字要与正则表达式里的名字相同。

3.无名分组是位置传参,有名分组是关键字传参。

4.无名分组与有名分组不能混合使用,单个可以重复使用。

反向解析

前言:html中a标签的href可以写网址的全称,也可以写后缀,写后缀会自动补全当前ip和port。

<a href='index'>点我跳转index</a>
<!--href='127.0.0.1:8000/index' -->

如上述标签可以跳转到127.0.0.1:8000/index,但是如果我路由匹配表达式出现了变化,那么这个地址就无法跳转到我想要的页面了,比如:

# 原本
url(r'^index/', views.index)
# 修改后
url(r'^index123/', views.index)

这个时候a标签就失效了,这该如何解决呢?反向解析就可以解决这个问题。

反向解析

通过反向解析可以获取到一个结果,该结果可以访问到一个路由。这么说可能有点模糊,我们来看实战:

第一步:给对应关系起别名

url(r'^index/', views.index, name='index_view')

第二步:使用模板语法修改a标签

<a href="{% url 'index_view' %}">点我跳转index</a>

这个时候随意修改路由的匹配表达式都会不出现a标签无法跳转。

后端也可以查看‘index_view’对应的路由:

from django.shortcuts import reverse
reverse('index_view') # 对应路由

根据这个路由,我们也可以进行重定向:

from django.shortcuts import reverse
def index(request):
return redirect(reverse('index_view'))

无名有名分组反向解析

无名分组反向解析

如果你在设置路由时使用了无名分组,那么你括号内的正则表达式的内容,需要人为指定。比如如下路由:

url(r'^index/(.*)', views.index, name='index_view')

这时候你的a标签需要指定内容,因为对于表达式 '^index/(.*)' 有多种方式匹配到,a标签不知道该使用哪个,所以需要人为指定。

比如我指定括号内的内容为:123

<a href="{% url 'index_view' 123 %}">点我跳转index</a>

后端也需要指定:

from django.shortcuts import reverse
reverse('index_view', args=(123,)) # /index/123

有名分组反向解析

如果你在设置路由时使用了有名分组,那么同理,需要人为指定内容。

路由:命名为address

url(r'^index/(?P<address>.*)', views.index, name='index_view')

a标签:设置address为123,两种方式都可

<a href="{% url 'index_view' 123 %}">点我跳转index</a>
<a href="{% url 'index_view' address=123 %}">点我跳转index</a>

后端:两种方式都可

from django.shortcuts import reverse
reverse('index_view', args=(123,)) # /index/123
reverse('index_view', kwargs=('address':123)) # /index/123

路由分发

如果一个django项目特别庞大,里面有很多应用,每个应用下有很多对应关系,这种情况如果把路由对应关系都写在项目同名文件夹下的urls.py文件下是明显不合理的,所以我们要把路由对应关系拆分开。

django支持每个应用都可以有自己独立的路由层、模板层、静态文件、视图层(默认)、模型层(默认)。

每个应用编写好自己的功能后,最后只需要整合一下就可以了。并且防止出现不同应用中出现相同路由,可以通过应用前缀区分,并且用到一个include方法:

总路由:整个应用app01与app02

from django.conf.urls import url, include
# 导入各个应用的路由,并起别名方便使用
from app01 import urls as app01_urls
from app02 import urls as app02_urls
urlpatterns = [
url(r'^app01/', include(app01_urls)),
url(r'^app02/', include(app02_urls)),
]

简写:

from django.conf.urls import url, include

urlpatterns = [
url(r'^app01/', include('app01.urls')),
url(r'^app02/', include('app02.urls')),
]

此时,访问应用app01下的index就需要这么输入:

http://127.0.0.1:8000/app01/index/

名称空间

不同的应用,设置路由时可能会使用相同的别名,比如:

应用app01路由:

urlpatterns = [
url(r'^index/', views.index, name='index_view'),
]

应用app02路由:

urlpatterns = [
url(r'^index/', views.index, name='index_view'),
]

a标签:

<a href="{% url 'index_view' %}">点我跳转index</a>
<a href="{% url 'index_view' %}">点我跳转index</a>

此时我就无法让我的a标签都可以跳到应用app01下index或者应用app02下index,它只能跳转到其中一个。

解决方法

方法一:总路由添加名称空间

urlpatterns = [
url(r'^app01/', include('app01.urls', namespace='app01')),
url(r'^app02/', include('app02.urls', namespace='app02')),
]

a标签:

<a href="{% url 'app01:index_view' %}">点我跳转index</a>
<a href="{% url 'app02:index_view' %}">点我跳转index</a>

方式二:应用路由起别名加上应用前缀,如

urlpatterns = [
url(r'^index/', views.index, name='app01_index_view'),
]

从根本解决问题。

django请求生命周期流程与路由层相关知识的更多相关文章

  1. Django框架10 /sweetalert插件、django事务和锁、中间件、django请求生命周期

    Django框架10 /sweetalert插件.django事务和锁.中间件.django请求生命周期 目录 Django框架10 /sweetalert插件.django事务和锁.中间件.djan ...

  2. Django框架深入了解_01(Django请求生命周期、开发模式、cbv源码分析、restful规范、跨域、drf的安装及源码初识)

    一.Django请求生命周期: 前端发出请求到后端,通过Django处理.响应返回给前端相关结果的过程 先进入实现了wsgi协议的web服务器--->进入django中间件--->路由f分 ...

  3. [Django框架 - 静态文件配置、request对象方法初识、 pycharm链接数据库、ORM实操增删改查、django请求生命周期]

    [Django框架 - 静态文件配置.request对象方法初识. pycharm链接数据库.ORM实操增删改查.django请求生命周期] 我们将html文件默认都放在templates文件夹下 将 ...

  4. django请求生命周期,FBV和CBV,ORM拾遗,Git

    一.django 请求生命周期 流程图: 1. 当用户在浏览器中输入url时,浏览器会生成请求头和请求体发给服务端请求头和请求体中会包含浏览器的动作(action),这个动作通常为get或者post, ...

  5. python 全栈开发,Day84(django请求生命周期,FBV和CBV,ORM拾遗,Git)

    一.django 请求生命周期 流程图: 1. 当用户在浏览器中输入url时,浏览器会生成请求头和请求体发给服务端请求头和请求体中会包含浏览器的动作(action),这个动作通常为get或者post, ...

  6. Django(35)Django请求生命周期分析(超详细)

    Django请求生命周期分析 1.客户端发送请求 在浏览器输入url地址,例如www.baidu.com,浏览器会自动补全协议(http),变为http://www.baidu.com,现在部分网站都 ...

  7. Django请求生命周期之响应内容

    Django请求生命周期: 1.发送http请求2.服务器接受,根据请求头中的url在路由关系表中进行匹配(从上到下)3.匹配成功后,执行指定的views函数 URL -> 函数 ==>F ...

  8. Django组件 - Django请求生命周期、中间件

    一.Django请求生命周期 在学习中间件之前,先了解一下Django的请求生命周期,如下图: 1)client代表浏览器,浏览器内部为我们封装了socket,Django的WSGI模块也封装了soc ...

  9. Django请求生命周期和ORM

    dajngo请求生命周期 django请求生命周期是:当用户在browser点击URL后,在django后台都发生了什么. 请求响应Http 1.发送Http请求 2.服务器接收,根据请求头中url在 ...

随机推荐

  1. 激光雷达 LOAM 论文 解析

    转自:https://blog.csdn.net/hltt3838/article/details/109261334 固态激光雷达的一段视频:https://v.qq.com/x/page/a078 ...

  2. 六个框架,一百多条检查项目,保证PCB设计不再出错

    一.资料输入阶段1.在流程上接收到的资料是否齐全(包括:原理图.*.brd文件.料单.PCB设计说明以及PCB设计或更改要求.标准化要求说明.工艺设计说明文件)2.确认PCB模板是最新的3. 确认模板 ...

  3. 如何监控微信小程序HTTP请求错误

    摘要: Fundebug的微信小程序错误监控插件更新至0.5.0,支持监控HTTP请求错误. 接入插件 接入Fundebug的错误监控插件非常简单,只需要下载fundebug.0.5.0.min.js ...

  4. Py的A+B

    程序会读入两行,每行都是一个数字,输出这两个数字的和 输入格式: 两行文字,每行都是一个数字 输出格式: 一行数字 输入样例: 18 21 输出样例: 39 代码: a = input() b = i ...

  5. 关于Symbol.iterator 学习笔记

    1.可以部署在对象上的一个遍历器 2. 遍历器是一个函数,需要返回一个含有一个next 方法的对象 const likeArray = {0:'a', 1: 'b', 2: 'c',3: 'd'. l ...

  6. Wireshark捕获网易云音乐音频文件地址

    打开Wireshark,开始捕获. 打开网易云音乐,然后播放一首歌. Wireshark停时捕获,然后在不活的文件中搜索字符串"mp3".可以发现有如下信息: 将其中的内容:&qu ...

  7. [ Vim ] 自动重载文件

    https://www.cnblogs.com/yeungchie/ 手动重载 :e 或者 :! 自动重载 set autoread 一般情况下,vim 切换缓冲区或者重新聚焦的时候会触发重载. 如果 ...

  8. 麒麟系统开发笔记(三):从Qt源码编译安装之编译安装Qt5.12

    前言   上一篇,是使用Qt提供的安装包安装的,有些场景需要使用到从源码编译的Qt,所以本篇如何在银河麒麟系统V4上编译Qt5.12源码.   银河麒麟V4版本   系统版本:   Qt源码下载    ...

  9. Springboot集成cache的key生成策略

    代码接上文:深度理解springboot集成redis缓存之源码解析 ## 1.使用SpEL表达式 @Cacheable(cacheNames = "emp",key = &quo ...

  10. MySQL启动过程详解二:核心模块启动 init_server_components()

    mysqld_main() 函数中,init_server_components() 函数负责MySQL核心模块的启动,包括mdl系统,Innodb存储引擎的启动等等: 1. mdl子系统初始化. 2 ...