django class-based views介绍与分析

       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服务的不应单独使用。

django自带的Mixins

通常情况下你并不需要直接用到这些Mixin,但你需要知道自己使用的Generic view继承了哪些Mixin,以便进行扩展。

·        SingleObjectMixin(单对象类)

·        MultipleObjectMixin(多对象类)

·        TemplateResponseMixin(模板响应类)

·        FormMixin(表单类)

·        YearMixin, MonthMixin,DayMixin, WeekMixin, DateMixin(几个基于时间关系的类)

django自带的generic views

   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初学者的建议是先从基础的学起。

你可能感兴趣的:(python)