Django的路由系统
Posted wjs521
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Django的路由系统相关的知识,希望对你有一定的参考价值。
URLconf配置
基本格式:
from django.conf.urls import url urlpatterns = [ url(正则表达式,views视图函数,参数,别名) ]
注意:Django2.0版本中的路由系统已经替换成下面的写法
from django.conf.urls import url urlpatterns = [ path(‘class/1994/‘,views.class_1994), path(‘class/<int:year>/<int:month>/‘,views.class_year_month), ]
参考说明:
1)正则表达式:一个正则表达式字符串
2)views视图函数:一个可调用对象,通常为一个视图函数或一个指定视图函数路径的字符串
3)参数:可选的要传递给视图函数的默认参数(字典形式)
4)别名:一个可选的name参数
正则表达式详解
基本配置
from django.conf.urls import url from . import views urlpatterns = [ url(r‘^class/([0-9]{4})/([0-9]{2})/([0-9]+)/$‘,views.class_detail) ]
注意事项:
1)urlpatterns中的元素从上往下逐一的去匹配正则表达式,一旦匹配成功则不在往下继续。
2)若要从URL中捕获一个值,则在它的周围放置一对圆括号(分组匹配)。
3)不需要添加一个前导的反斜杠,因为每个url都有。例如:^class而不是^/class。
4)每个正则表达式前面的‘r’是可选的但是建议加上。
补充说明:
# 是否开启URL访问地址后面不为/跳转至带有/的路径的配置顶 APPEND_SLASH = True
Django settings.py配置文件中默认没有APPEND_SLASH这个参数,但Django默认这个APPEND_SLASH = True。其作用就是在网址的末尾加“/”。
分组命名匹配
上面的示例使用简单的正则表达式分组匹配(通过圆括号)来捕获URL中的值并以位置参数形式传递给视图(views)。
在更高级的用法中,可以使用分组命名匹配的正则表达式组来平捕获URL中的值并以关键字参数形式传递给视图(views)。
在python的正则表达式中,分组命名正则表达式组的语法是(?P<name>pattern),其中name是组的名称,pattern是要匹配的模式。
from django.conf.urls import url from . import views urlpatterns = [ url(r‘^class/2003/$,views.class‘), url(r‘^student/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/$‘),views.year_month), url(r‘teacher/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/$‘),views.teacher_detail), ]
这个实现与前面的示例完全相同,只有一个细微的差别:捕获的值作为关键字参数而不是位置参数传递给视图函数。
针对url/teacher/2018/10/相当于按一下方式调用视图函数:
views.teacher_detail(request,year=‘2018‘,month=‘10‘)
在实际应用中,使用分组命名匹配的方式可以让你的URLconf更加明晰且不容易产生参数顺序问题的错误。
URLconf匹配的位置
URLconf在请求的URL上查找,将它当做一个普通的python字符串,不包括GET和POST参数以及域名。
在http:// www.example.com/myapp/?page=3请求中,URLconf将查找myapp/。
URLconf不检查请求的方法。换句话讲,所有的请求方法——同一URL的POST、GET、HEAD等等——都将路由到相同的函数。
捕获的参数永远都是字符串
每个在URLconf中捕获的参数都作为一个普通的python字符串传递给视图,无论正则表达式使用的是什么匹配方式。例如:下面这行URLconf中:
url(r‘^class/(?P<year>[0-9]{4})/$‘,views.year_class),
传递到视图函数views.year_class()中year参数永远是一个字符串类型。
视图函数中指定默认值
# urls.py中 from django.conf.urls import url from . import views urlpatterns = [ url(r‘^blog/$‘,views.page), url(r‘^blog/page(?P<num>[0-9]+)/$‘,views.page), ] # views.py中,可以为num指定默认值 def page(request,num="1"): pass
include其他URLconfs
from django.conf.urls import include,url urlpatterns = [ url(r‘^admin/‘,admin.site.urls), url(r‘^blog/‘,include(‘blog.urls‘)), # 可以包含其他的URLconfs文件 ]
命名URL和URL反向解析
可以给我们URL匹配规则起个名字,一个URL匹配模式起一个名字。
这样我们以后就不需要写死URL代码了,只需要通过名字来调用当前的URL。
简单的例子:
url(r"^home$",views.home,name="home"), # 给我的url匹配模式起名为home url(r‘^index/(d*)‘,views.index,name=‘index‘), # 给我的url匹配模式起名为index
在模板(可以理解为html文件)里面可以这样引用:
{% url "home" %}
在views函数中可以这样引用:
from django.urls import reverse redirect(reverse("index",args=("2018", )))
例子:
from django.conf.urls import url from . import views urlpatterns = [ url(r‘^(class|student|teacher)_list/$‘, views.list, name=‘list‘), ]
可以在模板的代码中使用下面的方法获得它们:
<li class="{% block class_active %}{% endblock %}"><a href="{% url ‘list‘ ‘class‘%}">班级列表</a></li> <li class="{% block student_active %}{% endblock %}"><a href="{% url ‘list‘ ‘student‘%}">学生列表</a></li> <li class="{% block teacher_active %}{% endblock %}"><a href="{% url ‘list‘ ‘teacher‘%}">老师列表</a></li>
在python代码中的使用:
def delete(request, table, del_id): table_Class = getattr(models, table.capitalize()) table_Class.objects.get(id=del_id).delete() return redirect(reverse(table)) 也可以加参数例如: return redirect(reverse(‘list‘, args=(id,))) # id传递一个id为了匹配成功。
命名空间模式
即使不同的APP使用相同的URL名称,URL的命名空间模式也可以让你唯一翻转命名的URL。
举个例子:
project中urls.py
from django.conf.urls import url,include urlpatterns = [ url(r‘^app01/‘,include(‘app01.urls‘,namespace=‘app01‘)), url(r‘^app02/‘,include(‘app02.urls‘,namespace=‘app02‘)), ]
app01中的urls.py
from django.conf.urls import url from app01 import views app_name = ‘app01‘ urlpatterns = [ url(r‘^(?P<year>d+)/$‘,views.detail,name=‘detail‘) ]
app02中的urls.py
from django.conf.urls import url from app02 import views app_name = ‘app02‘ urlpatterns = [ url(r‘^(?P<year>d+)/$‘,views.detail,name=‘detail‘) ]
两个APP中的url命名重复了,反转URL的时候就可以通过命名空间的名称得到当前的URL。
语法:
“命名空间名称:URL名称”
模板中使用:
{% url ‘app01:detail‘ year=1994 %}
views中的函数中使用
v = reverse(‘app01:detail‘,kwargs={‘year‘:1994})
这样就可以反转得到正确的URL了。
以上是关于Django的路由系统的主要内容,如果未能解决你的问题,请参考以下文章