django3-路由系统进阶
1.django的url到底是什么
就是路径 ,看成django的目录 ,每个目录对应个视图函数 ,当然一个url仅能对应一个函数
2.url的格式
url(正则表达式,views函数名,参数,别名)
这里的正则表达式有个问题就是可能会匹配到多个路径 ,匹配的方式是从上到下匹配即停 ,所以正则一定要严谨防止截胡
斜线' / '末尾加上会好很多,开头不用加
#urls.py from app01 import views #加载应用的地址函数模块 from app02 import views as app02views #这里一个问题就是不同应用的同名文件可以使用as来取别名 urlpatterns = [ url(r'^admin/$', admin.site.urls), url(r'^admin/add/', admin.site.urls), url(r'^admin/del/', admin.site.urls), url(正则表达式,views函数,参数,别名), ]
3.路由分发
场景: 当app过多所有的url都写在项目目录下的urls.py肯定是不行 ,所以要在每个app下都创建一个urls.py分发到各个app下即可 ,使用的是include函数做路由分发
#/项目目录下/urls.py
from django.conf.urls import url,include
from django.contrib import admin
from app1 import views
urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^login/', views.login),
url(r'^app1/', include('app1.urls')),
]
#/app1/urls.py
from django.conf.urls import url
from app1 import views
urlpatterns = [
url(r'^press/list', views.presslist),
url(r'^press/del', views.pressdel),
url(r'^press/add', views.pressadd),
url(r'^press/edit/', views.pressedit),
]
4.分组与命名分组
分组是在做什么? 我们传参可以使用get中的url携带参数 ,我们还能用分组直接获得参数传给视图函数 ,方便了不少呢
1).分组(肯定是正则的分组, 一组一个参数 ,参数仅仅能传递str类型)
#urls.py中
url(r'^press/del/(\d+)/', views.pressdel), #使用正则匹配一个分组
#views.py中
def pressdel(request, del_id): #接收url中的位置参数
models.presslist.objects.get(pk=del_id).delete()
return redirect('/app1/press/list')
#html中
<a>href="/app1/press/del/{{ obj.pk }}/"><i class="fa fa-remove" aria-hidden="true"></i></a> #点击时可以触发url
2).命名分组 ,和分组的区别就是使用的是关键字参数 ,可以不考虑顺序传递参数了
#urls.py中 url(r'^press/del/?P<del_id>(\d+)/', views.pressdel), #使用正则匹配一个分组 #views.py中 def pressdel(request, del_id): #接收url中的位置参数 models.presslist.objects.get(pk=del_id).delete() return redirect('/app1/press/list') #html中 <a>href="/app1/press/del/{{ obj.pk }}/"><i class="fa fa-remove" aria-hidden="true"></i></a> #点击时可以触发url
3).命名分组之默认值分组 ,默认值分组的意义是什么? 访问不带参数时的默认值 ,如下面的情况
#urls.py文件 url(r'^blog/page/$', views.blog), #访问首页没有参数的时候 ,视图函数中num=9999 url(r'^blog/page/(?P<num>[0-9]{1,2})', views.blog), #访问带页码的参数时 ,视图函数会使用分组中的num值 #views.py def blog(request,num='9999'):
5.路由的别名
场景: 重新规划了一下urls中的正则匹配地址 ,此时视图函数与模板中写死的跳转就都变了 ,那么改动实在太多了 ,使用路由别名! 每个url的正则无论如何变化 ,他的别名都不会变化
1).静态别名 ,就是url中没有携带参数(反向解析就是仅关注别名 ,别名后面如何不用管这就是反向解析)
视图函数中使用reverse()函数解析 ,模板中使用{% url '别名' %}解析
# urls.py文件 url(r'^halias/$', views.halias, name='hs'), url(r'^ralias/$', views.ralias, name='rs'), # views.py文件的redirect def halias(request): return render(request, 'test.html') def ralias(request): return redirect(reverse('rs')) # html文件 < a href = "{% url 'hs' %}" > 刷新操作 < / a >
2).动态别名 ,就是url中夹带了参数 ,反向解析的时候有点不同 ,其中分为参数为位置参数或者是关键字参数
####位置参数解析
#urls.py文件 url(r'^halias/([0-9]{1,2})/(..)/$', views.halias,name='hs'), #在views.py文件redirect(reverse) def halias(request,c1,c2): if c1 == '1': return redirect(reverse('hs',args=('34','op'))) return render(request,'test.html',{'c1':c1,'c2':c2}) #在html文件 <a href="{% url 'hs' '12' '12' %}">刷新操作</a>
####关键字参数解析
#urls.py文件 url(r'^halias/([0-9]{1,2})/(..)/$', views.halias,name='hs'), #在views.py文件redirect(reverse) def halias(request,c1,c2): if c1 == '1': return redirect(reverse('hs',kwargs={c1:'34',c2:'op'})) return render(request,'test.html',{'c1':c1,'c2':c2}) #在html文件 <a href="{% url 'hs' c1='12' c2='12' %}">刷新操作</a>
6.路由命名空间
场景: 反向解析虽然好用, 但是如果多个app下出现了重名的解析name模板与视图中怎么办?到底解析哪一个? 使用命名空间 ,在跟urls.py的文件中设定命名空间
#project/urls.py文件 urlpatterns = [ url(r'^app01/', include('app01.urls',namespace='ap1')), url(r'^app02/', include('app02.urls',namespace='ap2')), ] #html文件引入别名改动 <p> <a href="{% url 'ap1:ns' 参数1 参数2... %}">命名空间测试刷新当前页</a> </p> #views.py重定向使用别名时 .... return redirect(reverse('ap1:cbs',arg...)) ...
7.url相关规范
1)路由要在跟路由处完成分发 ,避免路由表过大
2)路由分发时标注命名空间 ,防止不同app下有重名路由
3)每个app下每条路由都要有别名 ,防止url变化引起大规模改动