Django中CBV视图的源码分析

前言:Django的视图处理方式有两种:

  • FBV(function base views) 是在视图里基于函数形式处理请求。
  • CBV(class base views)是在视图里基于类的形式处理请求。
Python是一个面向对象的编程语言,如果只用函数来开发,有很多面向对象的优点就错失了(继承、封装、多态)。所以Django在后来加入了Class-Based-View。可以让我们用类写View。这样做的优点主要下面两种:
  • 提高了代码的复用性,可以使用面向对象的技术,比如Mixin(多继承)
  • 可以用不同的函数针对不同的HTTP方法处理,而不是通过很多if判断,提高代码可读性

那下面我们来看一下django的源码,从而进一步了解Django的CBV

一、首先看url:

url:

url(r'register/$',views.Register.as_view())

django的CBV在url中的书写需要在view.类名后加.as_view(),从中可以了解到as_view一定是一个方法,这个方法加上了(),是在Django项目启动时就自动执行的,返回值也应该是 一个函数名,从而在匹配url后可以执行相应的函数

PS:关于分析源码的首要任务是需要清楚方法的调用者self的归属类!!!很重要
view.py:

class Register(View):

    def get(self,request):
        return render(request, 'register.html')

源码:

    @classonlymethod
    def as_view(cls, **initkwargs):
        pass
        def view(request, *args, **kwargs):
            self = cls(**initkwargs)
            if hasattr(self, 'get') and not hasattr(self, 'head'):
                self.head = self.get
            self.request = request
            self.args = args
            self.kwargs = kwargs
            return self.dispatch(request, *args, **kwargs)

        return view

源码执行简易流程图:


1.png

到此可以很明显的看出,Django项目启动后,CBV下的url执行后相当于:

url(r'register/$',views.Register.view)

二、接下来当有用户访问url时,对应url的view会执行对应请求方式下的类方法

源码:以下源码注意两点 1. cls涉及到了----《闭包作用域 》; 2. self的指向

    @classonlymethod
    def as_view(cls, **initkwargs):
        pass

        def view(request, *args, **kwargs):
            self = cls(**initkwargs)    //cls是Register类,self是实例化的对象
            if hasattr(self, 'get') and not hasattr(self, 'head'):
                self.head = self.get
            self.request = request
            self.args = args
            self.kwargs = kwargs
            return self.dispatch(request, *args, **kwargs)

        return view

view.py: 找dispatch方法时需要先找自己类中,没有再找继承类中

class Register(View):

    def get(self,request):
        return render(request, 'register.html'))

源码: 执行对应的dispatch方法

   def dispatch(self, request, *args, **kwargs):

        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
        return handler(request, *args, **kwargs)
7.png

到此Django基于CBV的源码执行流程就结束了,很好的利用了python面向对象的三大特性:继承、封装和多态。看完源码,是不是感觉python的面向对象有时候用起来意犹未尽。。。。。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明AI阅读 16,038评论 3 119
  • ​这一天,我们玩了泡泡袋画。 材料: 一个装快递的气泡袋,气泡一个一个凸起的圆形,印在纸上很有趣又好看。 颜料 一...
    妈咪私塾阅读 4,973评论 2 20
  • 悲伤与尽头 不知泪何物 眼前皆虚无 只求真实袭 无需酒解愁 无需被人怜 无需看世界 只求一角落 无泪乃悲凉 有泪却...
    然心江阅读 1,813评论 2 4
  • 前两天还吐槽庸俗化的极挑,差点没把我泪奔,时间这场回不去的旅途啊,希望我们都别把它看轻,你现在所拥有的,都是以后怀念的。
    丸子倩阅读 1,066评论 0 0