Django 第八课 -- 路由

目录

[一. 前言](#一. 前言)

[1.1. Django1.1.x 版本](#1.1. Django1.1.x 版本)

[1.2. Django 2.2.x 之后的版本](#1.2. Django 2.2.x 之后的版本)

[二. 正则路径中的分组](#二. 正则路径中的分组)

[2.1. 正则路径中的无名分组](#2.1. 正则路径中的无名分组)

[2.2. 正则路径中的有名分组](#2.2. 正则路径中的有名分组)

[三. 反向解析](#三. 反向解析)

[3.1. 普通路径](#3.1. 普通路径)

[3.2. 正则路径(无名分组)](#3.2. 正则路径(无名分组))

[3.3. 正则路径(有名分组)](#3.3. 正则路径(有名分组))

[四. 命名空间](#四. 命名空间)


一. 前言

路由简单的来说就是根据用户请求的 URL 链接来判断对应的处理程序,并返回处理结果,也就是 URL 与 Django 的视图建立映射关系。

Django 路由在 urls.py 配置,urls.py 中的每一条配置对应相应的处理方法。

Django 不同版本 urls.py 配置有点不一样:

1.1. Django1.1.x 版本

url() 方法:普通路径和正则路径均可使用,需要自己手动添加正则首位限制符号。

python 复制代码
from django.conf.urls import url # 用 url 需要引入

urlpatterns = [
    url(r'^admin/$', admin.site.urls),
    url(r'^index/$', views.index), # 普通路径
    url(r'^articles/([0-9]{4})/$', views.articles), # 正则路径
]

1.2. Django 2.2.x 之后的版本

  • path:用于普通路径,不需要自己手动添加正则首位限制符号,底层已经添加。
  • re_path:用于正则路径,需要自己手动添加正则首位限制符号。
python 复制代码
from django.urls import re_path # 用re_path 需要引入
urlpatterns = [
    path('admin/', admin.site.urls),
    path('index/', views.index), # 普通路径
    re_path(r'^articles/([0-9]{4})/$', views.articles), # 正则路径
]

**总结:**Django1.1.x 版本中的 url 和 Django 2.2.x 版本中的 re_path 用法相同。

二. 正则路径中的分组

2.1. 正则路径中的无名分组

无名分组按位置传参,一一对应。

views 中除了 request,其他形参的数量要与 urls 中的分组数量一致。

python 复制代码
urlpatterns = [
    path('admin/', admin.site.urls),
    re_path("^index/([0-9]{4})/$", views.index),
]
python 复制代码
from django.shortcuts import HttpResponse

def index(request, year):
    print(year) # 一个形参代表路径中一个分组的内容,按顺序匹配
    return HttpResponse('蜡笔小流')

2.2. 正则路径中的有名分组

语法:

python 复制代码
(?P<组名>正则表达式)

有名分组按关键字传参,与位置顺序无关。

views 中除了 request,其他形参的数量要与 urls 中的分组数量一致, 并且 views 中的形参名称要与 urls 中的组名对应。

python 复制代码
urlpatterns = [
    path('admin/', admin.site.urls),
    re_path("^index/(?P[0-9]{4})/(?P[0-9]{2})/$", views.index),
]
python 复制代码
from django.shortcuts import HttpResponse
def index(request, year, month):
    print(year,month) # 一个形参代表路径中一个分组的内容,按关键字对应匹配
    return HttpResponse('蜡笔小流')

路由分发(include)

存在问题:Django 项目里多个app目录共用一个 urls 容易造成混淆,后期维护也不方便。

解决:使用路由分发(include),让每个app目录都单独拥有自己的 urls。

步骤:

  • 1、在每个 app 目录里都创建一个 urls.py 文件。
  • 2、在项目名称目录下的 urls 文件里,统一将路径分发给各个 app 目录。
python 复制代码
from django.contrib import admin
from django.urls import path,include # 从 django.urls 引入 include
urlpatterns = [
    path('admin/', admin.site.urls),
    path("app01/", include("app01.urls")),
    path("app02/", include("app02.urls")),
]

在各自 app 目录下,写自己的 urls.py 文件,进行路径跳转。

app01 目录:

python 复制代码
from django.urls import path,re_path 
from app01 import views # 从自己的 app 目录引入 views 
urlpatterns = [ 
    re_path(r'^login/(?P<m>[0-9]{2})/$', views.index, ),
] 

app02 目录:

python 复制代码
from django.urls import path,re_path
from app02 import views # 从自己的 app 目录引入views 
urlpatterns = [ 
    re_path("^xxx/(?P[0-9]{4})/$", views.xxx), 
]

在各自 app 目录下的 views.py 文件中写各自的视图函数。

三. 反向解析

随着功能的增加,路由层的 url 发生变化,就需要去更改对应的视图层和模板层的 url,非常麻烦,不便维护。

这时我们可以利用反向解析,当路由层 url 发生改变,在视图层和模板层动态反向解析出更改后的 url,免去修改的操作。

反向解析一般用在模板中的超链接及视图中的重定向。

3.1. 普通路径

urls.py 中给路由起别名,name="路由别名"

python 复制代码
path("login1/", views.login, name="login")

views.py 中,从 django.urls 中引入 reverse,利用 reverse("路由别名") 反向解析:

python 复制代码
return redirect(reverse("login"))

在模板 templates 中的 HTML 文件中,利用 {% url "路由别名" %} 反向解析。

python 复制代码
<form action="{% url 'login' %}" method="post"> 

3.2. 正则路径(无名分组)

urls.py 中给路由起别名,name="路由别名"

python 复制代码
re_path(r"^login/([0-9]{2})/$", views.login, name="login")

views.py 中,从 django.urls 中引入 reverse,利用 reverse("路由别名",args=(符合正则匹配的参数,)) 反向解析。

python 复制代码
return redirect(reverse("login",args=(10,)))

在模板 templates 中的 HTML 文件中利用 {% url "路由别名" 符合正则匹配的参数 %} 反向解析。

python 复制代码
<form action="{% url 'login' 10 %}" method="post"> 

3.3. 正则路径(有名分组)

urls.py 中给路由起别名,name="路由别名"

python 复制代码
re_path(r"^login/(?P<year>[0-9]{4})/$", views.login, name="login")

views.py 中,从 django.urls 中引入 reverse,利用 **reverse("路由别名",kwargs={"分组名":符合正则匹配的参数})**反向解析。

python 复制代码
return redirect(reverse("login",kwargs={"year":3333}))

在模板 templates 中的 HTML 文件中,利用 {% url "路由别名" 分组名=符合正则匹配的参数 %} 反向解析。

python 复制代码
<form action="{% url 'login' year=3333 %}" method="post">

四. 命名空间

命名空间(英语:Namespace)是表示标识符的可见范围。

一个标识符可在多个命名空间中定义,它在不同命名空间中的含义是互不相干的。

一个新的命名空间中可定义任何标识符,它们不会与任何重复的标识符发生冲突,因为重复的定义都处于其它命名空间中。

**存在问题:**路由别名 name 没有作用域,Django 在反向解析 URL 时,会在项目全局顺序搜索,当查找到第一个路由别名 name 指定 URL 时,立即返回。当在不同的 app 目录下的urls 中定义相同的路由别名 name 时,可能会导致 URL 反向解析错误。

**解决:**使用命名空间。

普通路径

定义命名空间(include 里面是一个元组)格式如下:

python 复制代码
include(("app名称:urls","app名称"))

实例:

python 复制代码
path("app01/", include(("app01.urls","app01"))) 
path("app02/", include(("app02.urls","app02")))

在 app01/urls.py 中起相同的路由别名。

python 复制代码
path("login/", views.login, name="login")

views.py 中使用名称空间,语法格式如下:

python 复制代码
reverse("app名称:路由别名")

实例:

python 复制代码
return redirect(reverse("app01:login")

在 templates 模板的 HTML 文件中使用名称空间,语法格式如下:

python 复制代码
{% url "app名称:路由别名" %}

实例:

相关推荐
在下不上天6 分钟前
Flume日志采集系统的部署,实现flume负载均衡,flume故障恢复
大数据·开发语言·python
SEVEN-YEARS9 分钟前
深入理解TensorFlow中的形状处理函数
人工智能·python·tensorflow
EterNity_TiMe_14 分钟前
【论文复现】(CLIP)文本也能和图像配对
python·学习·算法·性能优化·数据分析·clip
Suyuoa25 分钟前
附录2-pytorch yolov5目标检测
python·深度学习·yolo
Iced_Sheep38 分钟前
干掉 if else 之策略模式
后端·设计模式
XINGTECODE1 小时前
海盗王集成网关和商城服务端功能golang版
开发语言·后端·golang
程序猿进阶1 小时前
堆外内存泄露排查经历
java·jvm·后端·面试·性能优化·oom·内存泄露
FIN技术铺1 小时前
Spring Boot框架Starter组件整理
java·spring boot·后端
好看资源平台1 小时前
网络爬虫——综合实战项目:多平台房源信息采集与分析系统
爬虫·python
凡人的AI工具箱2 小时前
15分钟学 Go 第 60 天 :综合项目展示 - 构建微服务电商平台(完整示例25000字)
开发语言·后端·微服务·架构·golang