路由系统
通过URL(Uniform Resource Locator,统一资源定位符)可以访问互联网上的资源——用户通过
浏览器向指定URL发起请求,Web服务器接收请求并返回用户请求的资源,因此可以将URL视为用户与服务器之间交互的桥梁。
在Django当中,路由系统的本质就是URL与要调用URL的视图函数的一个映射表,它的主要作用就
是让views里面的执行函数与请求的url建立映射关系,当请求来了的时候根据这个url映射来调用对应的
执行函数,从而返回给客户端相应的信息
Django处理请求的流程
当用户向你的Django站点请求一个页面时,系统会采用一个算法来确定要执行哪一段Python代码:
-
首先,Django会使用根路由解析模块(root URLconf)来解析路由。
-
Django加载该Python模块并查找变量 urlpatterns 。 它应该是 django.urls.path() 或者django.urls.re_path() 实例的Python列表。
-
Django按顺序遍历每个URL pattern,并在第一个匹配的请求URL被匹配时停下。
-
一旦某个URL pattern成功匹配,Django会导入并调用给定的视图,该视图是一个简单的Python
函数(或基于类的视图)。
这个视图会被传以以下参数:
一个 HttpRequest 的实例。
如果所匹配的正则表达式返回的是若干个无名组,那么该正则表达式所匹配的内容将被作为
位置参数提供给该视图。
关键字参数是由路径表达式匹配的任何指定部件组成的,在可选的 kwargs 参数中指定的任
何参数覆盖到 django.urls.path() 或 django.urls.re_path() 。
5.如果请求的URL没有匹配到任何一个表达式,或者在匹配过程的任何时刻抛出了一个异常,那么
Django 将调用适当的错误处理视图进行处理
URL配置
一个项目允许有多个urls.py,但Django需要一个urls.py作为入口,这个特殊的urls.py就是根
URLconf(根路由配置),它由settings.py文件中的ROOT_URLCONF指定
这条代码实在setting.py文件中
ROOT_URLCONF = 'mysite.urls'
以上示例通过ROOT_URLCONF指定了mysite目录下的urls.py作为根URLconf
为保证项目结构清晰,开发人员通常在Django项目的每个应用下创建urls.py文件,在其中为每个应用
配置子URL。路由系统接收到HTTP请求后,先根据请求的URL地址匹配根URLconf,找到匹配的子应用,再进一步匹配子URLconf,直到匹配完成
路由测试
1.创建新的项目
2.现将urls.py文件
3.views.py文件编写一个子应用
from django.shortcuts import render
from django.http import HttpResponse
# Create your views here.
def index(request):return HttpResponse('路由测试')
4.在子应用中的urls.py文件中创建子路由
from django.contrib import admin
from django.urls import path
from urlAPPS.views import indexurlpatterns = [path('index/', index),
]
5.在总项目的路由中添加上子路由
from django.contrib import admin
from django.urls import path,includeurlpatterns = [path('admin/', admin.site.urls),path('urlAPPS/',include('urlAPPS.urls')),
]
6.在views.py文件中添加新的子应用
def special_case_2022(request):return HttpResponse('路由测试1')def year_archive(request, year):return HttpResponse(f"year_archive{year}")def month_archive(request, year, month):return HttpResponse(f"month_archive 年:{year} 月:{month}")def article_detail(request, year, month, slug):return HttpResponse(f"article_detail 年:{year} 月:{month} {slug}")
7.在子路由中添加路由地址
path('articles/2022/', views.special_case_2022),path('articles/<int:year>/', views.year_archive),path('articles/<int:year>/<int:month>/', views.month_archive),path('articles/<int:year>/<int:month>/<slug:slug>/', views.article_detail),
路由转换器
Django2以上内置路由转换器可以显式地指定路由中参数的数据类型,Django中内置了5种路由转
换器,分别为str、int 、slug 、uuid 和path
str:匹配任何非空字符串,但不包含路由分隔符“/”。如果URL中没有指定参数类型,默认使用该
类型。
int:匹配0或任何正整数。并作为 int 返回。
slug:匹配由字母、数字、连字符和下划线组成的URL。例如, building-your-1st-django-site
uuid:匹配一个uuid格式的字符串。为了防止多个URL映射到同一页面中,该转换器必须包含连
字符,且所有字母均为小写。例如, 075194d3-6885-417e-a8a8-6c931e272f00 . 返回一个 UUID
实例
path:匹配包含路径分隔符 ‘/’ 在内的任意非空字符串。 相对于 str ,这允许你匹配一个完整的
URL路径,而不仅仅是URL路径的一部分。
from django.contrib import admin
from django.urls import path
from urlAPPS.views import index
from urlAPPS import viewsurlpatterns = [path('index/', index),path('articles/2022/', views.special_case_2022),path('articles/<int:year>/', views.year_archive),path('articles/<int:year>/<int:month>/', views.month_archive),path('articles/<int:year>/<int:month>/<slug:slug>/', views.article_detail),# str字符串类型测试path('articles/<str:year>/', views.str_cs),
]
正则表达式
如果路径和转换器语法不足以定义你的URL pattern,你还可以使用正则表达式。 为了使用正则表
达式,请使用 re_path() ,而不要使用 path() 。
在Python正则表达式中,命名正则表达式组的语法是 (?Ppattern) , 这里 name 是表达式组的名字
而 pattern 是要匹配的模式。
re_path(route, view, kwargs=None, name=None)
注意:
无论正则表达式匹配什么类型,每个捕获的参数都会以字符串形式传给视图。
正则表达式格式为:(?Ppattern),其中name表示分组名,pattern表示匹配的正则表达式。URL
匹配成功后,捕获到的参数会作为关键字参数传递给对应的视图,因此视图中的形式参数必须和正
则表达式中的分组名相同
若正则表达式只通过小括号“()”来捕获URL的参数,但未为其命名,则它是一个未命名正则表达
式,此时捕获的参数并将其以位置参数形式传递给对应视图
from django.contrib import admin
from django.urls import path , re_path
from urlAPPS.views import index
from urlAPPS import viewsurlpatterns = [# 路由转换器path('index/', index),# path('articles/2022/', views.special_case_2022),# path('articles/<int:year>/', views.year_archive),# path('articles/<int:year>/<int:month>/', views.month_archive),# path('articles/<int:year>/<int:month>/<slug:slug>/', views.article_detail),# str字符串类型测试# path('articles/<str:year>/', views.str_cs),# 正则表达式re_path('articles/2021/$', views.re_special_case_2022),re_path('articles/(?P<year>[0-9]{4})/$', views.re_year_archive),re_path('articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/$',views.re_month_archive),re_path('articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/(?P<slug>[-_\w]+) / $', views.re_article_detail),
]
向视图传递额外参数
path()函数、re_path()函数允许向视图传递额外参数,这些参数存放在一个字典类型的数据中,该
数据的键代表参数名,值代表参数值。re_path()函数与path()函数传递额外参数方式相同,以path()函数为例介绍如何向视图传递额外参数。
使用path()函数的第三个参数可以向视图传递额外参数
路由解析顺序
Django在接收到一个请求时,从主路由文件中的urlpatterns列表中以由上至下的顺序查找对应路
由规则,如果发现规则为include包含,则再进入被包含的urls中的urlpatterns列表由上至下进行
查询
反斜线问题
Django中定义路由时,通常以斜线/结尾,其好处是用户访问不以斜线/结尾的相同路径时,
Django会把用户重定向到以斜线/结尾的路径上,而不会返回404不存在
views.py
from django.shortcuts import render
from django.http import HttpResponse
# Create your views here.
def index(request):return HttpResponse('路由测试')def special_case_2022(request):return HttpResponse('路由测试1')def year_archive(request, year):return HttpResponse(f"year_archive{year}")def month_archive(request, year, month):return HttpResponse(f"month_archive 年:{year} 月:{month}")def article_detail(request, year, month, slug):return HttpResponse(f"article_detail 年:{year} 月:{month} {slug}")def str_cs(request,city):print('city的类型:',type(city))return HttpResponse(f'str_cs{city}')'''正则表达式'''
def re_special_case_2022(request):return HttpResponse('路由测试1')def re_year_archive(request, year):return HttpResponse(f"year_archive{year}")def re_month_archive(request, year, month):return HttpResponse(f"month_archive 年:{year} 月:{month}")def re_article_detail(request, year, month, slug):return HttpResponse(f"article_detail 年:{year} 月:{month} {slug}")'''传递额外参数'''
def blog_year_archive(request, year, major):return HttpResponse(f"blog_year_archive 年 {year} 额外参数:{major}")def re_blog_year_archive(request, year, major):return HttpResponse(f"blog_year_archive 年 {year} 额外参数:{major}")
urls.py
from django.contrib import admin
from django.urls import path , re_path
from urlAPPS.views import index
from urlAPPS import viewsurlpatterns = [# 路由转换器path('index/', index),# path('articles/2022/', views.special_case_2022),# path('articles/<int:year>/', views.year_archive),# path('articles/<int:year>/<int:month>/', views.month_archive),# path('articles/<int:year>/<int:month>/<slug:slug>/', views.article_detail),# str字符串类型测试# path('articles/<str:year>/', views.str_cs),# 正则表达式# re_path('articles/2021/$', views.re_special_case_2022),# re_path('articles/(?P<year>[0-9]{4})/$', views.re_year_archive),# re_path('articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/$',views.re_month_archive),# re_path('articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/(?P<slug>[-_\w]+) / $', views.re_article_detail),# 传递额外参数path('blog/<int:year>/', views.blog_year_archive, {'major': 'python'}),re_path('blog/(?P<year>[0-9]{4})/$', views.re_blog_year_archive, {'major':'python'}),
]