URLconf的技巧---------
第一种 每一个入口包括了它所关联的视图函数 直接传递一个函数对象 <-----------------------
from django.conf.urls.defaults import * from mysite.views import hello, current_datetime, hours_ahead <=============== urlpatterns = patterns('', (r'^hello/$', hello), (r'^time/$', current_datetime), (r'^time/plus/(\d{1,2})/$', hours_ahead), <============================= )
导入了views模块 <---------------------------------------------------------------
from django.conf.urls.defaults import * from mysite import views <============================== urlpatterns = patterns('', (r'^hello/$', views.hello ), (r'^time/$', views.current_datetime ), (r'^time/plus/(d{1,2})/$', views.hours_ahead ),<========================== )
传入一个包含模块名字和函数名的字符串,而不是函数对象本身<-----------------------------
from django.conf.urls.defaults import * urlpatterns = patterns('', (r'^hello/$', **'mysite.views.hello'** ), (r'^time/$', **'mysite.views.current_datetime'** ), (r'^time/plus/(d{1,2})/$', **'mysite.views.hours_ahead'** ),<============= )
提取出来公共前缀 <---------------------------------------------------------------
urlpatterns = patterns(**'mysite.views'** , <================================ (r'^hello/$', **'hello'** ), (r'^time/$', **'current_datetime'** ), (r'^time/plus/(d{1,2})/$', **'hours_ahead'** ), )
使用多个视图前缀 <---------------------------------------------------------------
from django.conf.urls.defaults import * urlpatterns = patterns('mysite.views', <============================ (r'^hello/$', 'hello'), (r'^time/$', 'current_datetime'), (r'^time/plus/(\d{1,2})/$', 'hours_ahead'), ) urlpatterns += patterns('weblog.views', <============================ (r'^tag/(\w+)/$', 'tag'), )
调试模式中的特例,动态构建urlpatterns 在django调试模式下修urlconf的行为
from django.conf import settings from django.conf.urls.defaults import * from mysite import views urlpatterns = patterns('', (r'^$', views.homepage), (r'^(\d{4})/([a-z]{3})/$', views.archive_month), ) if settings.DEBUG: <===================================== urlpatterns += patterns('', (r'^debuginfo/$', views.debug), )
使用命名组:
在Python的正则表达式中,命名的正则表达式组的语法是 (?P<name>pattern),这里的name是组的名字,pattern是某个匹配的模式。
无名组的例子:
from django.conf.urls.defaults import * from mysite import views urlpatterns = patterns('', (r'^articles/(\d{4})/$', views.year_archive), (r'^articles/(\d{4})/(\d{2})/$', views.month_archive), )
命名组的例子:相同的URLconf重写:
from django.conf.urls.defaults import * from mysite import views urlpatterns = patterns('', (r'^articles/(?P<year>\d{4})/$', views.year_archive), (r'^articles/(?P<year>\d{4})/(?P<month>\d{2})/$', views.month_archive), )
上边两段代码的功能完全一样,只有一个差别,取值前一个是以位置参数的方式传递给视图函数的,下一个是以关键字参数传递给视图函数的。
比如说,如果不带命名组 请求/articles/2012/08 等同于这样的函数调用:
month_archive(request, '2006', '03')
如果带了命名组就会变成这样的调用:
month_archive(request, year='2006', month='03')
使用命名组的话让URLconf看上去更加清晰,减少搞混参数次序的潜在bug,还可以让你在函数定义中对参数重新排序,即修改URL里提取参数的次序对视图没有影响。当然,代价是失去了简洁性,好处是可读性强。
URLconf解释器有关正则表达式中命名组和 非命名组所遵循的算法:
如果有任何命名的组,Django会忽略非命名组而直接使用命名组。
否则,Django会把所有非命名组以位置参数的形式传递。
在以上的两种情况,Django同时会以关键字参数的方式传递一些额外参数。 更具体的信息可参考下一节。
如果视图函数写的是十分类似的,只是有一点点的不同。比如说,有两个视图,内容是一致的,除了模板不同:
# urls.py from django.conf.urls.defaults import * from mysite import views urlpatterns = patterns('', (r'^foo/$', views.foo_view), (r'^bar/$', views.bar_view), <=========================== ) # views.py from django.shortcuts import render_to_response from mysite.models import MyModel def foo_view(request): m_list = MyModel.objects.filter(is_new=True) return render_to_response('template1.html', {'m_list': m_list}) def bar_view(request): m_list = MyModel.objects.filter(is_new=True) return render_to_response('template2.html', {'m_list': m_list})
代码做了重复的工作,在URL中用括号捕捉请求,然后在视图中检查决定使用哪个模板:
# urls.py from django.conf.urls.defaults import * from mysite import views urlpatterns = patterns('', (r'^(foo)/$', views.foobar_view), <========================== (r'^(bar)/$', views.foobar_view), ) # views.py from django.shortcuts import render_to_response from mysite.models import MyModel def foobar_view(request, url): m_list = MyModel.objects.filter(is_new=True) if url == 'foo': <=========================== template_name = 'template1.html' elif url == 'bar': template_name = 'template2.html' return render_to_response(template_name, {'m_list': m_list})
对于一个可选URL配置参数的优雅解决方法:URLconf中的每一个模式都可以包含第三个数据:一个关键字参数的字典:视图函数把字典制定的template_name当成另一个参数。
# urls.py from django.conf.urls.defaults import * from mysite import views urlpatterns = patterns('', (r'^foo/$', views.foobar_view, {'template_name': 'template1.html'}),<===== (r'^bar/$', views.foobar_view, {'template_name': 'template2.html'}), ) # views.py from django.shortcuts import render_to_response from mysite.models import MyModel def foobar_view(request, template_name): <==================== m_list = MyModel.objects.filter(is_new=True) return render_to_response(template_name, {'m_list': m_list})