请求生命周期流程图

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. 2022首场MASA技术团队黑客松赛事大赛完美落幕!精彩集锦

    Masa技术团队在2021年创立,这一年我们团队发布了我们第一个产品,Masa Blazor.登上了.NET Conf China,我们承诺,开源我们的产品,为开源社区增砖加瓦,一路上收获技术社区文章 ...

  2. Spark周总结(一)

    本周学习内容: 1.搭建虚拟机Spark环境 2.idea编写Scala脚本并在yarn上运行 总结: 这周是回家第一周,虽然没啥事,但是还是想放松放松,也是万事开头难,跟着教程做,但总有几步跟教程上 ...

  3. 百度图像识别SDK实验

    软件构造实验作业 实验名称:百度图像识别SDK实验 班级:信1905-1      学号:20194171      姓名:常金悦          一. 实验要求 每个步骤必须截图并说明 二.实验步 ...

  4. .NET程序设计实验二

    实验二  面向对象程序设计 一.实验目的 1. 理解类的定义.继承等面向对象的的基本概念: 2. 掌握C#语言定义类及其各种成员(字段,属性,方法)的方法: 3. 掌握方法覆盖的应用: 4. 掌握接口 ...

  5. CCF201604-2俄罗斯方块

    问题描述 俄罗斯方块是俄罗斯人阿列克谢·帕基特诺夫发明的一款休闲游戏. 游戏在一个15行10列的方格图上进行,方格图上的每一个格子可能已经放置了方块,或者没有放置方块.每一轮,都会有一个新的由4个小方 ...

  6. Input的校验表达式

    1.只是不能输入空格 <input type="text" onkeyup="this.value=this.value.replace(/^ +| +$/g,'' ...

  7. BootstrapBlazor实战 Tree树形控件使用(2)

    继续上篇实战BootstrapBlazor树型控件Tree内容, 本篇主要讲解整合Freesql orm快速制作数据库后台维护页面 demo演示的是Sqlite驱动,FreeSql支持多种数据库,My ...

  8. GIL全局解释器锁、协程运用、IO模型

    GIL全局解释器锁 一.什么是GIL 首先需要明确的一点是GIL并不是Python的特性,它是在实现Python解析器(CPython)时所引入的一个概念.就好比C是一套语言(语法)标准,但是可以用不 ...

  9. echarts饼图禁止鼠标悬浮高亮

    将高亮时的颜色和原本颜色手动设置成相同的值,把series.data里的itemStyle属性进行设置 代码如下: option = { color:['#3498db','#EEEEEE'], se ...

  10. [已解决] error: cannot convert `int*' to `int**' for argument `2' to `void print_f(int, int**)'

    #include "stdio.h" #include "stdlib.h" #include "time.h" void print_f( ...