在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 03 Python
进一步理解Python中的函数编程
Apr 13 Python
python数组过滤实现方法
Jul 27 Python
Python连接MySQL并使用fetchall()方法过滤特殊字符
Mar 13 Python
python的paramiko模块实现远程控制和传输示例
Oct 13 Python
Python+Turtle动态绘制一棵树实例分享
Jan 16 Python
Python 多线程不加锁分块读取文件的方法
Dec 11 Python
python 使用pandas计算累积求和的方法
Feb 08 Python
python实现控制COM口的示例
Jul 03 Python
手机使用python操作图片文件(pydroid3)过程详解
Sep 25 Python
Python unittest框架操作实例解析
Apr 13 Python
基于Python实现一个春节倒计时脚本
Jan 22 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面向对象的方法重载两种版本比较
2008/09/08 PHP
跟我学Laravel之安装Laravel
2014/10/15 PHP
php实现xml转换数组的方法示例
2017/02/03 PHP
Laravel 创建指定表 migrate的例子
2019/10/09 PHP
jquery中post方法用法实例
2014/10/21 Javascript
浅谈JavaScript超时调用和间歇调用
2015/08/30 Javascript
第一次接触神奇的Bootstrap基础排版
2016/07/26 Javascript
浅谈Angularjs link和compile的使用区别
2016/10/21 Javascript
JavaScript中省略元素对数组长度的影响
2016/10/26 Javascript
js返回顶部实例分享
2016/12/21 Javascript
jquery操作select取值赋值与设置选中实例
2017/02/28 Javascript
关闭Vue计算属性自带的缓存功能方法
2018/03/02 Javascript
解决vue v-for 遍历循环时key值报错的问题
2018/09/06 Javascript
Vue scrollBehavior 滚动行为实现后退页面显示在上次浏览的位置
2019/05/27 Javascript
基于JS实现table导出Excel并保留样式
2020/05/19 Javascript
[01:09:10]NB vs Liquid Supermajor小组赛 A组胜者组决赛 BO3 第一场 6.2
2018/06/04 DOTA
Python中函数的参数传递与可变长参数介绍
2015/06/30 Python
使用Python+Splinter自动刷新抢12306火车票
2018/01/03 Python
Python读取Json字典写入Excel表格的方法
2018/01/03 Python
Python使用matplotlib模块绘制图像并设置标题与坐标轴等信息示例
2018/05/04 Python
Python while循环使用else语句代码实例
2020/02/07 Python
一款纯css3实现的响应式导航
2014/10/31 HTML / CSS
简洁自适应404页面HTML好看的404源码
2020/12/16 HTML / CSS
Bergfreunde丹麦:登山装备网上零售商
2017/02/26 全球购物
自荐信怎么写好
2013/11/11 职场文书
材料工程专业毕业生求职信
2014/03/04 职场文书
校长竞聘演讲稿
2014/05/16 职场文书
药剂专业求职信
2014/06/20 职场文书
工地门卫岗位职责范本
2014/07/01 职场文书
2014年办公室文员工作总结
2014/11/12 职场文书
争先创优个人总结
2015/03/04 职场文书
写给老师的保证书
2015/05/09 职场文书
2015年秋季开学典礼校长致辞
2015/07/16 职场文书
先进个人事迹材料(2016推荐版)
2016/03/01 职场文书
python实现腾讯滑块验证码识别
2021/04/27 Python
yolov5返回坐标的方法实例
2022/03/17 Python