Django学习笔记之Class-Based-View


Posted in Python onFebruary 15, 2017

前言

大家都知道其实学习Django非常简单,几乎不用花什么精力就可以入门了。配置一个url,分给一个函数处理它,返回response,几乎都没有什么很难理解的地方。

写多了,有些问题才逐渐认识到。比如有一个view比较复杂,调用了很多其他的函数。想要把这些函数封装起来,怎么办?当然,可以用注释#------view------这样将函数隔离开,这种方法太low了,简直是在骗自己,连封装都算不上。

Python是一个面向对象的编程语言,如果只用函数来开发,有很多面向对象的优点就错失了(继承、封装、多态)。所以Django在后来加入了Class-Based-View。可以让我们用类写View。这样做的优点主要下面两种:

  1. 提高了代码的复用性,可以使用面向对象的技术,比如Mixin(多继承)
  2. 可以用不同的函数针对不同的HTTP方法处理,而不是通过很多if判断,提高代码可读性

使用class-based views

如果我们要写一个处理GET方法的view,用函数写的话是下面这样。

from django.http import HttpResponse
 
def my_view(request):
 if request.method == 'GET':
  # <view logic>
  return HttpResponse('result')

如果用class-based view写的话,就是下面这样。

from django.http import HttpResponse
from django.views import View
 
class MyView(View):
 def get(self, request):
  # <view logic>
  return HttpResponse('result')

Django的url是将一个请求分配给可调用的函数的,而不是一个class。针对这个问题,class-based view提供了一个as_view()静态方法(也就是类方法),调用这个方法,会创建一个类的实例,然后通过实例调用dispatch()方法,dispatch()方法会根据request的method的不同调用相应的方法来处理request(如get() , post()等)。到这里,这些方法和function-based view差不多了,要接收request,得到一个response返回。如果方法没有定义,会抛出HttpResponseNotAllowed异常。

在url中,就这么写:

# urls.py
from django.conf.urls import url
from myapp.views import MyView
 
urlpatterns = [
 url(r'^about/$', MyView.as_view()),
]

类的属性可以通过两种方法设置,第一种是常见的Python的方法,可以被子类覆盖。

from django.http import HttpResponse
from django.views import View
 
class GreetingView(View):
 greeting = "Good Day"
 
 def get(self, request):
  return HttpResponse(self.greeting)
 
# You can override that in a subclass
 
class MorningGreetingView(GreetingView):
 greeting = "Morning to ya"

第二种方法,你也可以在url中指定类的属性:

在url中设置类的属性Python

urlpatterns = [
 url(r'^about/$', GreetingView.as_view(greeting="G'day")),
]

使用Mixin

我觉得要理解django的class-based-view(以下简称cbv),首先要明白django引入cbv的目的是什么。在django1.3之前,generic view也就是所谓的通用视图,使用的是function-based-view(fbv),亦即基于函数的视图。有人认为fbv比cbv更pythonic,窃以为不然。python的一大重要的特性就是面向对象。而cbv更能体现python的面向对象。cbv是通过class的方式来实现视图方法的。class相对于function,更能利用多态的特定,因此更容易从宏观层面上将项目内的比较通用的功能抽象出来。关于多态,不多解释,有兴趣的同学自己Google。总之可以理解为一个东西具有多种形态(的特性)。cbv的实现原理通过看django的源码就很容易明白,大体就是由url路由到这个cbv之后,通过cbv内部的dispatch方法进行分发,将get请求分发给cbv.get方法处理,将post请求分发给cbv.post方法处理,其他方法类似。怎么利用多态呢?cbv里引入了mixin的概念。Mixin就是写好了的一些基础类,然后通过不同的Mixin组合成为最终想要的类。

所以,理解cbv的基础是,理解Mixin。Django中使用Mixin来重用代码,一个View Class可以继承多个Mixin,但是只能继承一个View(包括View的子类),推荐把View写在最右边,多个Mixin写在左边。Mixin也是比较复杂的技术,本文不详细说了,以后写一篇针对Mixin的文章吧。

使用装饰器

在CBV中,可以使用method_decorator来装饰方法。

from django.contrib.auth.decorators import login_required
from django.utils.decorators import method_decorator
from django.views.generic import TemplateView
 
class ProtectedView(TemplateView):
 template_name = 'secret.html'
 
 @method_decorator(login_required)
 def dispatch(self, *args, **kwargs):
  return super(ProtectedView, self).dispatch(*args, **kwargs)

也可以写在类上面,传入方法的名字。

@method_decorator(login_required, name='dispatch')
class ProtectedView(TemplateView):
 template_name = 'secret.html'

如果有多个装饰器装饰一个方法,可以写成一个list。例如,下面这两种写法是等价的。

decorators = [never_cache, login_required]
 
@method_decorator(decorators, name='dispatch')
class ProtectedView(TemplateView):
 template_name = 'secret.html'
 
@method_decorator(never_cache, name='dispatch')
@method_decorator(login_required, name='dispatch')
class ProtectedView(TemplateView):
 template_name = 'secret.html'

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流。

Python 相关文章推荐
python中map、any、all函数用法分析
Apr 21 Python
Linux系统上Nginx+Python的web.py与Django框架环境
Dec 25 Python
Python实战小程序利用matplotlib模块画图代码分享
Dec 09 Python
78行Python代码实现现微信撤回消息功能
Jul 26 Python
Python3 jupyter notebook 服务器搭建过程
Nov 30 Python
python实现点击按钮修改数据的方法
Jul 17 Python
pygame库实现俄罗斯方块小游戏
Oct 29 Python
解决Pytorch 加载训练好的模型 遇到的error问题
Jan 10 Python
python统计字符的个数代码实例
Feb 07 Python
Python基于百度AI实现OCR文字识别
Apr 02 Python
python tkinter实现定时关机
Apr 21 Python
Python语言规范之Pylint的详细用法
Jun 24 Python
关于Python面向对象编程的知识点总结
Feb 14 #Python
Django日志模块logging的配置详解
Feb 14 #Python
深入理解Python对Json的解析
Feb 14 #Python
Python 2与Python 3版本和编码的对比
Feb 14 #Python
利用Python爬虫给孩子起个好名字
Feb 14 #Python
python 出现SyntaxError: non-keyword arg after keyword arg错误解决办法
Feb 14 #Python
让python 3支持mysqldb的解决方法
Feb 14 #Python
You might like
利用PHP扩展vld查看PHP opcode操作步骤
2013/03/04 PHP
php之Smarty模板使用方法示例详解
2014/07/08 PHP
php图片的二进制转换实现方法
2014/12/15 PHP
php实现分页显示
2015/11/03 PHP
解决 firefox 不支持 document.all的方法
2007/03/12 Javascript
Javascript 各浏览器的 Javascript 效率对比
2008/01/23 Javascript
DEDECMS如何为文章添加HOT NEW标志图片
2015/08/14 Javascript
JS实现网页上随滚动条滚动的层效果代码
2015/11/04 Javascript
jquery文字填写自动高度的实现方法
2016/11/07 Javascript
NodeJs下的测试框架Mocha的简单介绍
2017/02/22 NodeJs
在node.js中怎么屏蔽掉favicon.ico的请求
2017/03/01 Javascript
初识NodeJS服务端开发入门(Express+MySQL)
2017/04/07 NodeJs
AngularJS使用ng-repeat遍历二维数组元素的方法详解
2017/11/11 Javascript
jquery+css3实现熊猫tv导航代码分享
2018/02/12 jQuery
vue3修改link标签默认icon无效问题详解
2019/10/09 Javascript
在vue中created、mounted等方法使用小结
2020/07/21 Javascript
vue深度监听(监听对象和数组的改变)与立即执行监听实例
2020/09/04 Javascript
[01:35]2018年度CS GO最佳战队-完美盛典
2018/12/17 DOTA
跟老齐学Python之不要红头文件(1)
2014/09/28 Python
Python3学习笔记之列表方法示例详解
2017/10/06 Python
Python爬虫信息输入及页面的切换方法
2018/05/11 Python
Numpy数据类型转换astype,dtype的方法
2018/06/09 Python
mac下给python3安装requests库和scrapy库的实例
2018/06/13 Python
django+echart数据动态显示的例子
2019/08/12 Python
利用pandas合并多个excel的方法示例
2019/10/10 Python
如何将anaconda安装配置的mmdetection环境离线拷贝到另一台电脑
2020/10/15 Python
python实现不同数据库间数据同步功能
2021/02/25 Python
Smashbox英国官网:美国知名彩妆品牌
2017/11/13 全球购物
企业厂长岗位职责
2013/12/17 职场文书
开展批评与自我批评发言材料
2014/05/15 职场文书
政府领导干部个人对照检查材料思想汇报
2014/09/24 职场文书
股权转让协议范本
2014/12/07 职场文书
先进个人事迹材料范文
2014/12/30 职场文书
大二学年个人总结
2015/03/03 职场文书
2016年敬老月活动总结
2016/04/05 职场文书
2019暑期安全倡议书!
2019/06/27 职场文书