Django的CBV与FBV

Django的CBV与FBV

 

FBV

FBV(function base views) 就是在视图里使用函数处理请求。

在之前django的学习中,我们一直使用的是这种方式,所以不再赘述。

CBV

CBV(class base views) 就是在视图里使用类处理请求。

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':
            return HttpResponse('OK')

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

复制代码
from django.http import HttpResponse
from django.views import View
  
class MyView(View):
def get(self, request): return HttpResponse('OK')
复制代码

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'^index/$', MyView.as_view()),
]
复制代码

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

复制代码
from django.http import HttpResponse
from django.views import View
  
class GreetingView(View):
    name = "yuan"
    def get(self, request):
         return HttpResponse(self.name)
  
# You can override that in a subclass
  
class MorningGreetingView(GreetingView):
    name= "alex"
复制代码

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

在url中设置类的属性Python

urlpatterns = [
   url(r'^index/$', GreetingView.as_view(name="egon")),
]

使用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写在左边。

 

CBV使用配置

路径url的配置

cbv 顾名知义就是通过类的方法来调用,我们在url中配置为如下路径

 url(r'^cbv.html/', views.Cbv.as_view()),

这里的Cbv是一个class 类,要想使用cbv方法,这个路径后面还得必须有一个as_view()这个是必须的固定格式

views里面函数的格式

在views里面配置类,需要导入一个模块

from django.views.generic import View #这个是导入的模块,原来的django版本从django.views 里面可以直接导入View,但是现在需要加一个generic才可以
class Cbv(View): #这里必须要继承View这个类,只有继承了这个url那里的as_view()才会有这个方法 def get(self,request): return HttpResponse('cbv-get') def post(self,request): return HttpResponse('cbv-post')
浏览器get方式访问

Django的CBV方式讲解

创建一个login登陆页面测试post方法
views配置
from django.views.generic import View
class Cbv(View): def get(self,request): # return HttpResponse('cbv-get') return render(request,'login.html') #发送到login.html def post(self,request): return HttpResponse('cbv-post')

login的页面配置代码

<!DOCTYPE html>
<html lang="en"> <head> <meta charset="UTF-8"> <title>cbv学习</title> </head> <body> <form action="/cbv.html/" method="post"> <input type="text" name="username"> <input type="submit" value="提交"> </form> </body> </html>
浏览器访问查看点击提交后的结果

Django的CBV方式讲解

点击提交
Django的CBV方式讲解

这里通过查看View的源码,可以看到里面会有很多种提交方法
http_method_names = ['get', 'post', 'put', 'patch', 'delete', 'head', 'options', 'trace']
使用ajax的时候这些方法都是可以使用的。

另外继承类不光有View,还有很多的,查看源码就可以看到的
Django的CBV方式讲解

我的django版本号是
C:\Users\Tony>python3 -m django --version
1.9.13

cbv匹配原理

这种更具url来匹配方法的是通过反射方法(getattr)来做的。请求过来后先走dispatch这个方法,这个方法存在View类中。

    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 return handler(request, *args, **kwargs)
定制dispatch

如果需要批量对方法,例如get,post等方法做一些操作的时候,这里我们可以手动写一个dispatch,这个dispatch就和装饰器的效果一样。因为请求来的时候总是先走的dispatch。

from django.views.generic import View
class Cbv(View): def dispatch(self, request, *args, **kwargs): print('操作前的操作') obj = super(Cbv,self).dispatch(request, *args, **kwargs) print('操作后的操作代码') return obj def get(self,request): # return HttpResponse('cbv-get') return render(request,'login.html') def post(self,request): return HttpResponse('cbv-post')

这次我们在通过浏览器访问的时候,发现不管get或者post方法,都会走print代码,
Django的CBV方式讲解

 

使用cbv实现视图

 1 from django.views import View
 2 from django.utils.decorators import method_decorator
 3 
 4 
 5 class LoginView(View):
 6 
 7     def get(self, request):
 8         return render(request, "login.html")
 9 
10     def post(self, request):
11         user = request.POST.get("name")
12         pwd = request.POST.get("pwd")
13         if user == "safly" and pwd == "123":
14             # 登陆成功
15             # 写session
16             request.session["user2"] = user
17             request.session.set_expiry(5)
18             return redirect("/index/")
19 
20 def index(request):
21     return render(request,"index.html")
View Code

CBV结合装饰器

直接加在视图类上,但method_decorator必须传 name 关键字参数
 1 from django.views import View
 2 from django.utils.decorators import method_decorator
 3 
 4 
 5 def wrapper(func):
 6     @wraps(func)
 7     def inner(request, *args, **kwargs):
 8         # 登录校验
 9         cookie_k = request.session.get("user01", None)
10         if cookie_k:
11             # 表示已经登录的用户
12             ret = func(request, *args, **kwargs)
13             return ret
14         else:
15             # 滚去登录
16             return redirect("/login/")
17 
18     return inner
19 
20 
21 class LoginView(View):
22 
23     def get(self, request):
24         return render(request, "login.html")
25 
26     def post(self, request):
27         user = request.POST.get("name")
28         pwd = request.POST.get("pwd")
29         if user == "safly" and pwd == "123":
30             # 登陆成功
31             # 写session
32             request.session["user01"] = user
33             # request.session.set_expiry(5)
34             return redirect("/index/")
35 
36 
37 @method_decorator(wrapper, name="get")
38 class IndexView(View):
39     def get(self, request):
40         user = request.session.get("user01", "游客")
41         return render(request, "index.html", {"user": user})
View Code
* 加在CBV视图的get或post方法上*
# @method_decorator(wrapper, name="get")
class IndexView(View):
    @method_decorator(wrapper)
    def get(self, request):
        user = request.session.get("user02", "游客")
        return render(request, "index.html", {"user": user})
View Code
* 加在dispatch方法上*
 1 # @method_decorator(wrapper, name="get")
 2 class IndexView(View):
 3     ## 这么写所有的请求方法都要做登录校验
 4     @method_decorator(wrapper)
 5     def dispatch(self, request, *args, **kwargs):
 6         return super(IndexView,self).dispatch(request,*args,**kwargs)
 7 
 8     # @method_decorator(wrapper)
 9     def get(self, request):
10         user = request.session.get("user04", "游客")
11         return render(request, "index.html", {"user": user})
View Code

CSRF Token

CSRF Token相关装饰器在CBV只能加到dispatch方法上

备注:

csrf_protect,为当前函数强制设置防跨站请求伪造功能,即便settings中没有设置全局中间件。
csrf_exempt,取消当前函数防跨站请求伪造功能,即便settings中设置了全局中间件
 1 from django.views.decorators.csrf import csrf_exempt, csrf_protect
 2 
 3 
 4 class HomeView(View):
 5 
 6     @method_decorator(csrf_exempt)
 7     def dispatch(self, request, *args, **kwargs):
 8         return super(HomeView, self).dispatch(request, *args, **kwargs)
 9 
10     def get(self, request):
11         return render(request, "home.html")
12 
13     def post(self, request):
14         print("Home View POST method...")
15         return redirect("/index/")
View Code

 

转载于:https://www.cnblogs.com/lbzbky/articles/10920770.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值