class-based views是django从1.3版本引入的一种新的特性,就是使用类的形式来描述view。class-based views对django带来的不只是编程方法的改变,而是程序设计哲学的改变。Django在最开始的设计的时候利用函数的方式来实现view,这种方式也是利用了python的动态语言特性,可以说是python’s 的编程方式。现在引入了class-based views的方式,很多人认为他太过于复杂,并且这种方式是no python’s。我对于这些说法是持保留态度的,首先python是一种面向对象的语言,利用class-based views的方式可以充分利用面向对象的优点,我在这里要强调的一点是python他不只是一种脚本语言,他完全可以构建大型的应用,并且这已经是事实(如openstack)。class-based views实际起到的作用不是使程序变得复杂了,恰恰相反他使程序的变得简单。他可以将一些公用的东西抽象出来,供别的类继承使用,减少了大量的重复的单调的冗余的代码编写工作。
class-based views的复杂只是存在于去接受他理解他,刚开始使用可能觉得有些不适应,等慢慢对他有了深入的理解就会慢慢体会到他的好处。对于class-based views的学习其实也不是很复杂,虽然中文资料很少,但是官方文档还是有较为详细的介绍。我认为最好的学习方法还是去读他的相关的源代码。其实他本身的源码量不是很多,如果硬着头皮去读完全可以读懂的。
各种class-based views的实现都是通过继承view这个基类然后再mixin各种实现相关功能的mixin class。Mixin是利用python的多继承的特性,这些Mixin class都是为view服务的不应单独使用。
通常情况下你并不需要直接用到这些Mixin,但你需要知道自己使用的Generic view继承了哪些Mixin,以便进行扩展。
· SingleObjectMixin(单对象类)
· MultipleObjectMixin(多对象类)
· TemplateResponseMixin(模板响应类)
· FormMixin(表单类)
· YearMixin, MonthMixin,DayMixin, WeekMixin, DateMixin(几个基于时间关系的类)
TemplateView渲染指定的模板
RedirectView重定向
DetailView 察看对象详细信息
ListView 显示对象列表
Editing views 实现编辑
FormView 显示Form
CreateView 创建对象
UpdateView 更新对象
DeleteView 删除对象
http://akonizo.com/ 中有详细的类视图。
在实现自己的可重用views时,可从下面的类继承,并可以通过增加自定义Mixin的方式进行功能扩展,openstack的horizon就是通过继承generic.TemplateView实现了了几种自定义的view类型,然后被各实际页面继承使用。
View中最重要的两个方法就是get和post,他们实际对应http的get和post方法,在执行http的get或post请求时被实际调用。as_view非常重要他是处理实际的 request-response 过程的接入点,在urls.py中实际调用 url(r'^$',TemplateListView.as_view(), name='poject-list'),了as_view方法
@classonlymethod
def as_view(cls, **initkwargs):
"""
Main entry point for a request-response process.
"""
# sanitize keyword arguments
for key in initkwargs:
if key in cls.http_method_names:
raise TypeError(u"You tried to pass in the %s method name as a "
u"keyword argument to %s(). Don't do that."
% (key, cls.__name__))
if not hasattr(cls, key):
raise TypeError(u"%s() received an invalid keyword %r" % (
cls.__name__, key))
def view(request, *args, **kwargs):
self = cls(**initkwargs)
if hasattr(self, 'get') and not hasattr(self, 'head'):
self.head = self.get
return self.dispatch(request, *args, **kwargs)
# take name and docstring from class
update_wrapper(view, cls, updated=())
# and possible attributes set by decorators
# like csrf_exempt from dispatch
update_wrapper(view, cls.dispatch, assigned=())
return view
dispatch方法时实际调用前边所说的post,get方法
def dispatch(self, request, *args, **kwargs):
# Try to dispatch to the right method; if a method doesn't exist,
# defer to the error handler. Also defer to the error handler if the
# request method isn't on the approved list.
if request.method.lower() in self.http_method_names:
handler = getattr(self, request.method.lower(), self.http_method_not_allowed)
else:
handler = self.http_method_not_allowed
self.request = request
self.args = args
self.kwargs = kwargs
return handler(request, *args, **kwargs)
这里利用了python的动态语言的一个比较常见的技巧,用handler=getattr(self, request.method.lower(), self.http_method_not_allowed)得到方法名,handler(request, *args, **kwargs)实际执行方法,handler的值可能是post或get,实现了http的名字和view中方法名对应。其实熟悉http的朋友可能都知道,不止有post,get方法还有 'put', 'delete', 'head', 'options', 'trace'等方法。但是class view中了post,get方法比较常用就以这连个方法作为论述的例子,当然你可以根据需求实现put,delete等方法原理是和post,和get完全一样的。
读懂class view的前提是你对django的,先关基础知识比较熟悉。所以对于django初学者的建议是先从基础的学起。