在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记录详细调用堆栈日志的方法
May 05 Python
简介Django中内置的一些中间件
Jul 24 Python
简单解决Python文件中文编码问题
Nov 22 Python
python实现人人自动回复、抢沙发功能
Jun 08 Python
在Python中获取两数相除的商和余数方法
Nov 10 Python
python合并已经存在的sheet数据到新sheet的方法
Dec 11 Python
python redis 批量设置过期key过程解析
Nov 26 Python
pytorch forward两个参数实例
Jan 17 Python
基于梯度爆炸的解决方法:clip gradient
Feb 04 Python
windows python3安装Jupyter Notebooks教程
Apr 13 Python
Jupyter notebook无法导入第三方模块的解决方式
Apr 15 Python
python获取字符串中的email
Mar 31 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数组添加与删除单元的常用函数实例分析
2015/02/16 PHP
PHP根据树的前序遍历和中序遍历构造树并输出后序遍历的方法
2017/11/10 PHP
JavaScript起点(严格模式深度了解)
2013/01/28 Javascript
js清理Word格式示例代码
2014/02/13 Javascript
使用JavaScript实现连续滚动字幕效果的方法
2015/07/07 Javascript
JS实现网页右侧带动画效果的伸缩窗口代码
2015/10/29 Javascript
jquery获取复选框checkbox的值的简单实现方法
2016/05/26 Javascript
Bootstrap编写导航栏和登陆框
2016/05/30 Javascript
一个仿微博登陆邮箱提示框js开发案例
2016/07/28 Javascript
ztree实现左边动态生成树右边为内容详情功能
2017/11/03 Javascript
angularjs 页面自适应高度的方法
2018/01/17 Javascript
Angular利用trackBy提升性能的方法
2018/01/26 Javascript
jQuery插件Validation表单验证详解
2018/05/26 jQuery
vue 实现cli3.0中使用proxy进行代理转发
2019/10/30 Javascript
JS实现字体背景跑马灯
2020/01/06 Javascript
[59:48]DOTA2-DPC中国联赛 正赛 VG vs Magma BO3 第一场 1月26日
2021/03/11 DOTA
python基础教程之循环介绍
2014/08/29 Python
利用QT写一个极简单的图形化Python闹钟程序
2015/04/07 Python
Python ftp上传文件
2016/02/13 Python
python使用tensorflow保存、加载和使用模型的方法
2018/01/31 Python
python操作excel的包(openpyxl、xlsxwriter)
2018/06/11 Python
Python定义一个跨越多行的字符串的多种方法小结
2018/07/19 Python
tensorflow 打印内存中的变量方法
2018/07/30 Python
Python 正则表达式匹配字符串中的http链接方法
2018/12/25 Python
Python 图像处理: 生成二维高斯分布蒙版的实例
2019/07/04 Python
python如何爬取网站数据并进行数据可视化
2019/07/08 Python
python 上下文管理器及自定义原理解析
2019/11/19 Python
bonprix匈牙利:女士、男士和儿童服装
2019/07/19 全球购物
法学个人求职信范文
2014/01/27 职场文书
施工材料员岗位职责
2014/02/12 职场文书
大四学生找工作的自荐信
2014/03/27 职场文书
小学生期末评语大全
2014/04/21 职场文书
医院营销工作计划
2015/01/16 职场文书
2015年党支部书记工作总结
2015/05/21 职场文书
旗帜观后感
2015/06/08 职场文书
创业计划书之酒厂
2019/10/14 职场文书