在Django中创建URLconf相关的通用视图的方法


Posted in Python onJuly 20, 2015

抽取出我们代码中共性的东西是一个很好的编程习惯。 比如,像以下的两个Python函数:

def say_hello(person_name):
  print 'Hello, %s' % person_name

def say_goodbye(person_name):
  print 'Goodbye, %s' % person_name

我们可以把问候语提取出来变成一个参数:

def greet(person_name, greeting):
  print '%s, %s' % (greeting, person_name)

通过使用额外的URLconf参数,你可以把同样的思想应用到Django的视图中。

了解这个以后,你可以开始创作高抽象的视图。 更具体地说,比如这个视图显示一系列的 Event 对象,那个视图显示一系列的 BlogEntry 对象,并意识到它们都是一个用来显示一系列对象的视图的特例,而对象的类型其实就是一个变量。

以这段代码作为例子:

# urls.py

from django.conf.urls.defaults import *
from mysite import views

urlpatterns = patterns('',
  (r'^events/$', views.event_list),
  (r'^blog/entries/$', views.entry_list),
)

# views.py

from django.shortcuts import render_to_response
from mysite.models import Event, BlogEntry

def event_list(request):
  obj_list = Event.objects.all()
  return render_to_response('mysite/event_list.html', {'event_list': obj_list})

def entry_list(request):
  obj_list = BlogEntry.objects.all()
  return render_to_response('mysite/blogentry_list.html', {'entry_list': obj_list})

这两个视图做的事情实质上是一样的: 显示一系列的对象。 让我们把它们显示的对象的类型抽象出来:

# urls.py

from django.conf.urls.defaults import *
from mysite import models, views

urlpatterns = patterns('',
  (r'^events/$', views.object_list, {'model': models.Event}),
  (r'^blog/entries/$', views.object_list, {'model': models.BlogEntry}),
)

# views.py

from django.shortcuts import render_to_response

def object_list(request, model):
  obj_list = model.objects.all()
  template_name = 'mysite/%s_list.html' % model.__name__.lower()
  return render_to_response(template_name, {'object_list': obj_list})

就这样小小的改动,我们突然发现我们有了一个可复用的,模型无关的视图! 从现在开始,当我们需要一个视图来显示一系列的对象时,我们可以简简单单的重用这一个 object_list 视图,而无须另外写视图代码了。 以下是我们做过的事情:

    我们通过 model 参数直接传递了模型类。 额外URLconf参数的字典是可以传递任何类型的对象,而不仅仅只是字符串。

    这一行: model.objects.all() 是 鸭子界定 (原文:

    我们使用 model.__name__.lower() 来决定模板的名字。 每个Python的类都有一个 __name__ 属性返回类名。 这特性在当我们直到运行时刻才知道对象类型的这种情况下很有用。 比如, BlogEntry 类的 __name__ 就是字符串 'BlogEntry' 。

    这个例子与前面的例子稍有不同,我们传递了一个通用的变量名给模板。 当然我们可以轻易的把这个变量名改成 blogentry_list 或者 event_list ,不过我们打算把这当作练习留给读者。

因为数据库驱动的网站都有一些通用的模式,Django提供了一个通用视图的集合,使用它可以节省你的时间。 我们将会在下一章讲讲Django的内置通用视图。
提供视图配置选项

如果你发布一个Django的应用,你的用户可能会希望配置上能有些自由度。 这种情况下,为你认为用户可能希望改变的配置选项添加一些钩子到你的视图中会是一个很好的主意。 你可以用额外URLconf参数实现。

一个应用中比较常见的可供配置代码是模板名字:

def my_view(request, template_name):
  var = do_something()
  return render_to_response(template_name, {'var': var})

了解捕捉值和额外参数之间的优先级 额外的选项

当冲突出现的时候,额外URLconf参数优先于捕捉值。 也就是说,如果URLconf捕捉到的一个命名组变量和一个额外URLconf参数包含的变量同名时,额外URLconf参数的值会被使用。

例如,下面这个URLconf:

from django.conf.urls.defaults import *
from mysite import views

urlpatterns = patterns('',
  (r'^mydata/(?P<id>\d+)/$', views.my_view, {'id': 3}),
)

这里,正则表达式和额外字典都包含了一个 id 。硬编码的(额外字典的) id 将优先使用。 就是说任何请求(比如, /mydata/2/ 或者 /mydata/432432/ )都会作 id 设置为 3 对待,不管URL里面能捕捉到什么样的值。

聪明的读者会发现在这种情况下,在正则表达式里面写上捕捉是浪费时间的,因为 id 的值总是会被字典中的值覆盖。 没错,我们说这个的目的只是为了让你不要犯这样的错误。

Python 相关文章推荐
初学python数组的处理代码
Jan 04 Python
Python实现从URL地址提取文件名的方法
May 15 Python
python安装PIL模块时Unable to find vcvarsall.bat错误的解决方法
Sep 19 Python
python 表达式和语句及for、while循环练习实例
Jul 07 Python
Python基于pandas实现json格式转换成dataframe的方法
Jun 22 Python
[原创]Python入门教程4. 元组基本操作
Oct 31 Python
利用Pycharm断点调试Python程序的方法
Nov 29 Python
python画图系列之个性化显示x轴区段文字的实例
Dec 13 Python
python如果快速判断数字奇数偶数
Nov 13 Python
Python3基于plotly模块保存图片表格
Aug 03 Python
python3跳出一个循环的实例操作
Aug 18 Python
Python日志打印里logging.getLogger源码分析详解
Jan 17 Python
python通过socket查询whois的方法
Jul 18 #Python
Python字符串匹配算法KMP实例
Jul 18 #Python
Python通过正则表达式选取callback的方法
Jul 18 #Python
Django的URLconf中使用缺省视图参数的方法
Jul 18 #Python
Python的Django框架中URLconf相关的一些技巧整理
Jul 18 #Python
在Django框架中伪造捕捉到的URLconf值的方法
Jul 18 #Python
Django中传递参数到URLconf的视图函数中的方法
Jul 18 #Python
You might like
处理php自动反斜杠的函数代码
2010/01/05 PHP
PHP打开和关闭文件操作函数总结
2014/11/18 PHP
详解PHP的Yii框架中组件行为的属性注入和方法注入
2016/03/18 PHP
php中array_unshift()修改数组key注意事项分析
2016/05/16 PHP
centos下file_put_contents()无法写入文件的原因及解决方法
2017/04/01 PHP
利用PHP获取汉字首字母并且分组排序详解
2017/10/22 PHP
PHP与Web页面交互操作实例分析
2020/06/02 PHP
PHP http请求超时问题解决方案
2020/11/13 PHP
实现51Map地图接口(示例代码)
2013/11/22 Javascript
轻松创建nodejs服务器(7):阻塞操作的实现
2014/12/18 NodeJs
实现非常简单的js双向数据绑定
2015/11/06 Javascript
深入解析JavaScript编程中的this关键字使用
2015/11/09 Javascript
微信小程序 form组件详解
2016/10/25 Javascript
jQuery插件ContextMenu自定义图标
2017/03/15 Javascript
node.js的http.createServer过程深入解析
2019/06/06 Javascript
Angular8路由守卫原理和使用方法
2019/08/29 Javascript
[01:08:32]DOTA2-DPC中国联赛 正赛 DLG vs PHOENIX BO3 第二场 1月18日
2021/03/11 DOTA
简单了解Python下用于监视文件系统的pyinotify包
2015/11/13 Python
python+opencv实现动态物体追踪
2018/01/09 Python
用python代码将tiff图片存储到jpg的方法
2018/12/04 Python
PyQt5实现简单数据标注工具
2019/03/18 Python
django ManyToManyField多对多关系的实例详解
2019/08/09 Python
Django中使用MySQL5.5的教程
2019/12/18 Python
python+selenium+chrome实现淘宝购物车秒杀自动结算
2021/01/07 Python
Expedia韩国官网:亚洲发展最快的在线旅游门户网站
2018/02/26 全球购物
Mio Skincare法国官网:身体紧致及孕期身体护理
2018/04/04 全球购物
戴森英国官网:Dyson英国
2019/05/07 全球购物
Python是如何进行类型转换的
2013/06/09 面试题
捐书寄语赠言
2014/01/18 职场文书
目标责任书格式
2014/07/28 职场文书
对党的十八届四中全会的期盼
2014/10/17 职场文书
护理心得体会范文
2016/01/22 职场文书
2016年机关单位节能宣传周活动总结
2016/04/05 职场文书
Nginx tp3.2.3 404问题解决方案
2021/03/31 Servers
动画「进击的巨人」第86话播出感谢绘公开
2022/03/21 日漫
openEuler 搭建java开发环境的详细过程
2022/06/10 Servers