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")),

 CBV应用装饰器

1.指定方法上添加装饰器

from django.views import View
from django.utils.decorators import method_decorator

#装饰器
def wrapper(func):
    def inner(*args,**kwargs):
        return func(*args,**kwargs)
    return inner

class Foo(View):

    @method_decorator(wrapper)
    def get(self,request):
        pass

    def post(self,request):
        pass

2.在类上添加

通过name指定类里面的哪个方法加上装饰器。

from django.views import View
from django.utils.decorators import method_decorator

#装饰器
def wrapper(func):
    def inner(*args,**kwargs):
        return func(*args,**kwargs)
    return inner

@method_decorator(wrapper,name='get')
class Foo(View):

    def get(self,request):
        pass

    def post(self,request):
        pass

 若name='dispatch',则类Foo里面的所有方法都加上了装饰器

from django.views import View
from django.utils.decorators import method_decorator

#装饰器
def wrapper(func):
    def inner(*args,**kwargs):
        return func(*args,**kwargs)
    return inner

@method_decorator(wrapper,name='dispatch')  #get与post都引用装饰器
class Foo(View):

    def get(self,request):
        pass

    def post(self,request):
        pass

 

posted on 2019-04-27 20:23  xufat  阅读(136)  评论(0)    收藏  举报

导航

/* 返回顶部代码 */ TOP