Django的路由系统
URL配置(URLconf)就像Django 所支撑网站的目录。它的本质是URL与要为该URL调用的视图函数之间的映射表;你就是以这种方式告诉Django,对于这个URL调用这段代码,对于那个URL调用那段代码。
基本格式
''' urlpatterns = [ url(正则表达式, views视图函数,参数,别名), ] 参数说明: 一个正则表达式字符串 一个可调用对象,通常为一个视图函数或一个指定视图函数路径的字符串 可选的要传递给视图函数的默认参数(字典形式) 一个可选的name参数 '''
注意:
Django 2.0版本中的路由系统已经替换成下面写法(官方文当):
from django.urls import path urlpatterns = [ path('articles/2003/', views.special_case_2003), 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), ]
注意事项
- urlpatterns中的元素按照书写顺序从上往下逐一匹配正则表达式,一旦匹配成功则不再继续。
- 若要从URL中捕获一个值,只需要在它周围放置一对圆括号(分组匹配)。
- 不需要添加一个前导的反斜杠,因为每个URL 都有。例如,应该是^articles 而不是 ^/articles。
- 每个正则表达式前面的'r' 是可选的但是建议加上。
关键点
# 是否开启URL访问地址后面不为/跳转至带有/的路径的配置项 APPEND_SLASH=True
分组命名匹配
上面的示例使用简单的正则表达式分组匹配(通过圆括号)来捕获URL中的值并以位置参数形式传递给视图。 在更高级的用法中,可以使用分组命名匹配的正则表达式组来捕获URL中的值并以关键字参数形式传递给视图。 在Python的正则表达式中,分组命名正则表达式组的语法是(?P<name>pattern),其中name是组的名称,pattern是要匹配的模式。 下面是以上URLconf 使用命名组的重写:
URLconf匹配的位置
URLconf 在请求的URL 上查找,将它当做一个普通的Python 字符串。不包括GET和POST参数以及域名。 例如,http://www.example.com/myapp/ 请求中,URLconf 将查找myapp/。 在http://www.example.com/myapp/?page=3 请求中,URLconf 仍将查找myapp/。 URLconf 不检查请求的方法。换句话讲,所有的请求方法 —— 同一个URL的POST、GET、HEAD等等 —— 都将路由到相同的函数。
捕获的参数永远都是字符串
每个在URLconf中捕获的参数都作为一个普通的Python字符串传递给视图,无论正则表达式使用的是什么匹配方式。例如,下面这行URLconf 中:
url(r'^articles/(?P<year>[0-9]{4})/$', views.year_archive),
传递到视图函数views.year_archive()
中的year
参数永远是一个字符串类型。
命名URL和URL反向解析
url的反向解析
如上述,url中的正则表达式即为请求的路径,根据路径匹配执行对应的视图函数,如下,当我们请求路径符合正则表达式要求,就会执行对应的login函数:
其中login函数为:
login.html文件为:
根据以上我们所写的代码逻辑,当通过get请求/login/路径时,我们执行符合此请求路径对应的login()函数,函数渲染一个login.htl的登录页面,当用户验证完信息点击提交按钮时,再次以post形式请求/login/对应的login()函数。
但是,在某些情况下,我们需要更改请求路径的正则表达式,即更换一个请求路径,这时候我们就需要对上述的url中的正则表达式和login.html中的提交路径同时进行更改成我们想要的,但是当我们有很多个文件和位置都有请求/login/的时候,我们一个个的手动的更改显然不是很合理。这就需要用到反向解析,也就是url中别名参数的作用,具体实例如下:以上即称为反向解析,无论url中请求路径的如何更改,只要别名不改,程序中对此函数的请求都可以通过别名反向解析的形式找到。
以上即称为反向解析,无论url中请求路径的如何更改,只要别名不改,程序中对此函数的请求都可以通过别名反向解析的形式找到。
二、Including other URLconfs
django项目中创建应用的命令为
python manage.py startapp app01(app01为项目名)
在django项目创建后,会有一个url.py文件,用于映射不同请求路径对应的函数,但是当我们一个django项目中有很多应用的话,各个项目的url的映射关系均放在此文件中显然不妥,不利于代码间的解耦。此时需要在此url文件中设置Including other URLconfs,设置后各应用对应的url映射关系就可以放在应用自己目录下urls.py文件中。如下:
此情况访问应用下路径映射函数的的方式如下例所示:
三、正则字符串参数
1、无名分组
通过在正则字符中使用一对括号,就可以实现分组,并可以获得分组的参数,如下例:
url(r'^articles/(d{4})/(d{2})/$', views.articles2),
#articles2(request,组1参数,组2参数)
当请求路径为:http://127.0.0.1:8000/app01/articles/2004/12/,路径符合上述正则表达式字符串,执行articles2(request,2014,12)函数,即要求定义articles2函数时候必须要接收三个参数,映射过程是按照位置进行传参,函数实例如下:
def articles2(request,year,month): return HttpResponse(year+"---"+month)
2、有名分组
上面的示例使用简单的、没有命名的正则表达式组(通过圆括号)来捕获URL 中的值并以位置 参数传递给视图。在更高级的用法中,可以使用命名的正则表达式组来捕获URL 中的值并以关键字 参数传递给视图。在Python 正则表达式中,命名正则表达式组的语法是(?P<name>pattern)
,其中name
是组的名称,pattern
是要匹配的模式。
url(r'^articles/(?P<y>d{4})/(?P<m>d{2})/$', views.articles3)
#articles3(request,y=对应参数,m=对应参数)
当请求路径为:http://127.0.0.1:8000/app01/articles/2004/12/,路径符合上述正则表达式字符串,执行articles3(request,y=2014,m=12)函数,即要求定义articles3函数时必须要接收三个参数,且参数必须为y和m,映射过程是按照关键字进行传参,函数实例如下:
def articles3(request,y,m):
#参数y和m必须与分组名一致,因为是按照关键字传参,位置随意 return HttpResponse(y+"-----"+m)
四、redirect函数
redirect()函数也称为重定向函数,现在以第一部分中登陆实例为讲述其执行过程:
第一次网页请求: 请求url: http://127.0.0.1:8000/login/ 请求类型:GET 无请求数据 login/-------> views.login------>login() 响应到一个login.html页面 第二次按钮提交: 请求url: http://127.0.0.1:8000/login.html/ 请求类型:POST 有请求数据 {"user":"alex","pwd":"123"} login.html/-------> views.login------>login() 响应的return redirect("/index/")到浏览器,通知浏览器再发送请求:"/index/" 请求url: http://127.0.0.1:8000/index/ 请求类型:GET 没请求数据 index/ --->url(r'^index/', views.index),---->index() 响应一个index.html
有上述过程分析可知,重定向函数redirect在重定向的过程中实际是接受到两次响应,第一次通过响应告诉浏览器重新发送请求到需要重定向的路径,第二次是浏览器发送重定向路径请求后得到响应结果。
namespace(反向生成)
例如:
urlpatterns = 【 url(r'^admin/',admin.site.urls) , url (r'^ dealer/,include('dealer.urls','dealer,'de')), # de-->namespace前缀 url (r'^ jassin/,include('jassin.urls','jassin','ja')), # ja-->namespace前缀 】
反向生成需要加前缀
2、防止app名字相同
总结
总结成5个知识点 1、普通 /index/ def index(request):。。。 2、正则 /index/(d+)/ def index(request,nid):..... 3、name /index/xxx/xxx/xxx/xxx/xxxx/xxx name=“n1” url = reverse('n1') # /index/xxx/xxx/xxx/xxx/xxxx/xxx /index/xxx/xxx/xxx/xxx/xxxx/xxx/(d+)/ name="n2" url = reverse('n2',args=(11,)) # /index/xxx/xxx/xxx/xxx/xxxx/xxx/11/ url = reverse('n2',args=(22,)) # /index/xxx/xxx/xxx/xxx/xxxx/xxx/22/ /index/xxx/xxx/xxx/xxx/xxxx/xxx/(?P<nid>d+)/ name='n3' # /index/xxx/xxx/xxx/xxx/xxxx/xxx/22/ 4、路由分发(include) 5、namespace(反向生成)