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将字符串转换成数组的方法
Apr 29 Python
对python csv模块配置分隔符和引用符详解
Dec 12 Python
Python使用字典的嵌套功能详解
Feb 27 Python
windows系统中Python多版本与jupyter notebook使用虚拟环境的过程
May 15 Python
利用selenium爬虫抓取数据的基础教程
Jun 10 Python
python实现关闭第三方窗口的方法
Jun 28 Python
pyinstaller参数介绍以及总结详解
Jul 12 Python
Python猴子补丁知识点总结
Jan 05 Python
Python pip安装第三方库实现过程解析
Jul 09 Python
深入了解Python enumerate和zip
Jul 16 Python
python数字转对应中文的方法总结
Aug 02 Python
Python如何快速找到多个字典中的公共键(key)
Apr 29 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
php5中date()得出的时间为什么不是当前时间的解决方法
2008/06/30 PHP
PHP网页游戏学习之Xnova(ogame)源码解读(七)
2014/06/23 PHP
thinkphp浏览历史功能实现方法
2014/10/29 PHP
PHP实现文件下载详解
2014/11/27 PHP
smarty简单应用实例
2015/11/03 PHP
Yii2中OAuth扩展及QQ互联登录实现方法
2016/05/16 PHP
基于jquery的simpleValidate简易验证插件
2014/01/31 Javascript
js设置cookie过期当前时间减去一秒相当于立即过期
2014/09/04 Javascript
jQuery中scrollLeft()方法用法实例
2015/01/16 Javascript
JavaScript图片轮播代码分享
2015/07/31 Javascript
JavaScript获取服务器端时间的方法
2016/11/29 Javascript
微信通过页面(H5)直接打开本地app的解决方法
2017/09/09 Javascript
把JavaScript代码改成ES6语法不完全指南(分享)
2017/09/10 Javascript
浅谈如何使用 webpack 优化资源
2017/10/20 Javascript
vuex的module模块用法示例
2018/11/12 Javascript
浅谈react-router@4.0 使用方法和源码分析
2019/06/04 Javascript
JavaScript setTimeout()基本用法有哪些
2020/11/04 Javascript
Vue——解决报错 Computed property &quot;****&quot; was assigned to but it has no setter.
2020/12/19 Vue.js
[59:53]DOTA2-DPC中国联赛 正赛 VG vs Elephant BO3 第二场 3月6日
2021/03/11 DOTA
python 随机数生成的代码的详细分析
2011/05/15 Python
flask + pymysql操作Mysql数据库的实例
2017/11/13 Python
Python爬取当当、京东、亚马逊图书信息代码实例
2017/12/09 Python
Python解决抛小球问题 求小球下落经历的距离之和示例
2018/02/01 Python
python快速建立超简单的web服务器的实现方法
2018/02/17 Python
Python登录注册验证功能实现
2018/06/18 Python
django2.0扩展用户字段示例
2019/02/13 Python
使用pyqt5 tablewidget 单元格设置正则表达式
2019/12/13 Python
python 实现让字典的value 成为列表
2019/12/16 Python
Python3爬虫中Ajax的用法
2020/07/10 Python
Django实现微信小程序支付的示例代码
2020/09/03 Python
python drf各类组件的用法和作用
2021/01/12 Python
pycharm 关闭search everywhere的解决操作
2021/01/15 Python
CSS3实现粒子旋转伸缩加载动画
2016/04/22 HTML / CSS
机关作风整顿个人剖析材料
2014/10/06 职场文书
接收函
2019/04/22 职场文书
python实现简单的聊天小程序
2021/07/07 Python