Django中传递参数到URLconf的视图函数中的方法
有时你会发现你写的视图函数是十分类似的,只有一点点的不同。比如说,你有两个视图,它们的内容是一致的,除了它们所用的模板不太一样:
#urls.py fromdjango.conf.urls.defaultsimport* frommysiteimportviews urlpatterns=patterns('', (r'^foo/$',views.foo_view), (r'^bar/$',views.bar_view), ) #views.py fromdjango.shortcutsimportrender_to_response frommysite.modelsimportMyModel deffoo_view(request): m_list=MyModel.objects.filter(is_new=True) returnrender_to_response('template1.html',{'m_list':m_list}) defbar_view(request): m_list=MyModel.objects.filter(is_new=True) returnrender_to_response('template2.html',{'m_list':m_list})
我们在这代码里面做了重复的工作,不够简练。起初你可能会想,通过对两个URL都使用同样的视图,在URL中使用括号捕捉请求,然后在视图中检查并决定使用哪个模板来去除代码的冗余,就像这样:
#urls.py fromdjango.conf.urls.defaultsimport* frommysiteimportviews urlpatterns=patterns('', (r'^(foo)/$',views.foobar_view), (r'^(bar)/$',views.foobar_view), ) #views.py fromdjango.shortcutsimportrender_to_response frommysite.modelsimportMyModel deffoobar_view(request,url): m_list=MyModel.objects.filter(is_new=True) ifurl=='foo': template_name='template1.html' elifurl=='bar': template_name='template2.html' returnrender_to_response(template_name,{'m_list':m_list})
这种解决方案的问题还是老缺点,就是把你的URL耦合进你的代码里面了。如果你打算把/foo/改成/fooey/的话,那么你就得记住要去改变视图里面的代码。
对一个可选URL配置参数的优雅解决方法:URLconf里面的每一个模式都可以包含第三个数据:一个关键字参数的字典:
有了这个概念以后,我们就可以把我们现在的例子改写成这样:
#urls.py fromdjango.conf.urls.defaultsimport* frommysiteimportviews urlpatterns=patterns('', (r'^foo/$',views.foobar_view,{'template_name':'template1.html'}), (r'^bar/$',views.foobar_view,{'template_name':'template2.html'}), ) #views.py fromdjango.shortcutsimportrender_to_response frommysite.modelsimportMyModel deffoobar_view(request,template_name): m_list=MyModel.objects.filter(is_new=True) returnrender_to_response(template_name,{'m_list':m_list})
如你所见,这个例子中,URLconf指定了template_name。而视图函数会把它当成另一个参数。
这种使用额外的URLconf参数的技术以最小的代价给你提供了向视图函数传递额外信息的一个好方法。