Django ------ 高级 view 和 URLconf 配置 URLconf的技巧

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})

 

posted @ 2012-08-30 11:12  事件轮询,回不到过去  阅读(325)  评论(0编辑  收藏  举报