FBV
FBV(function base views) 就是在視圖里使用函數處理請求。
在之前django的學習中,我們一直使用的是這種方式,所以不再贅述。
CBV
CBV(class base views) 就是在視圖里使用類處理請求。
Python是一個面向對象的編程語言,如果只用函數來開發,有很多面向對象的優點就錯失了(繼承、封裝、多態)。所以Django在后來加入了Class-Based-View。可以讓我們用類寫View。這樣做的優點主要下面兩種:
- 提高了代碼的復用性,可以使用面向對象的技術,比如Mixin(多繼承)
- 可以用不同的函數針對不同的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和FBV代碼示例
FBV
fbv就是在url中一個路徑對應一個函數
urlpatterns = [ url(r'^admin/', admin.site.urls), url(r'^index/', views.index) ]
在視圖函數中
def index(request): return render(request, 'index.html')
CBV
cbv就是在url中一個路徑對應一個類
urlpatterns = [ url(r'^admin/', admin.site.urls), url(r'^index/', views.IndexView.as_view()) # 執行類后面的as_view()方法,是父類里面的方法 ]
在視圖函數中
from django.views import View class IndexView(View): # 以get形式訪問會執行get函數,一般情況下獲取數據 def get(self, *args, **kwargs): return HttpResponse('666') # 以post形式訪問的話會執行post函數,一般情況下發送數據 def post(self, *args, **kwargs): return HttpResponse('999')
注意:
- cbv定義類的時候必須要繼承view
- 在寫url的時候必須要加as_view
- 類里面使用form表單提交的話只有get和post方法
- 類里面使用ajax發送數據的話支持定義以下很多方法
restful規范:
'get'獲取數據, 'post'創建新數據, 'put'更新, 'patch'局部更新, 'delete'刪除, 'head', 'options', 'trace'
CBV重新定義dispatch函數
所有的方法本質上都是通過dispatch這個函數反射執行,如果想要在執行get或post方法前執行其他步驟,可以重寫dispatch
class IndexView(View): # 重寫父類的dispatch方法,如果不重寫,他會執行父類的dispatch方法, def dispatch(self, request, *args, **kwargs): print('before') res = super(IndexView, self).dispatch(request, *args, **kwargs) print('after') return res # 以get形式訪問會執行get函數 def get(self, *args, **kwargs): return HttpResponse('666') # 以post形式訪問的話會執行post函數 def post(self, *args, **kwargs): return HttpResponse('999')
下面我們根據上面的寫法添加用戶登錄驗證
from django.shortcuts import render, HttpResponse, redirect from django.views import View class LoginView(View): def get(self, request): return render(request, 'login.html') def post(self, request): user = request.POST.get('user') pwd = request.POST.get('pwd') if user == 'rdw' and pwd == '666': request.session['user_info'] = 'rdw' return redirect('/index/') else: return render(request, 'login.html') class IndexView(View): # 重寫父類的dispatch方法,如果不重寫,他會執行父類的dispatch方法, def dispatch(self, request, *args, **kwargs): if not request.session.get('user_info'): return redirect('/login/') res = super(IndexView, self).dispatch(request, *args, **kwargs) return res # 以get形式訪問會執行get函數 def get(self, request, *args, **kwargs): return render(request, 'index.html') # 以post形式訪問的話會執行post函數 def post(self, *args, **kwargs): return HttpResponse('999')
給視圖類添加裝飾器
如果有多個程序需要用戶登錄驗證的話會造成代碼冗余,可以使用繼承很好的解決這個問題,但是還有更好的方法,那就是基於裝飾器實現登錄驗證
定義裝飾器
def login_test(func): def inner(request, *args, **kwargs): if not request.session.get('user_info'): return redirect('/login/') return func(*args, **kwargs) return inner
1、直接添加在dispatch里面,這樣每個函數都會執行
from django.utils.decorators import method_decorator
@method_decorator(login_test)
def dispatch(self, request, *args, **kwargs):
res = super(IndexView, self).dispatch(request, *args, **kwargs)
return res
2、添加在每一個函數中
from django.utils.decorators import method_decorator
@method_decorator(login_test)
def get(self, request, *args, **kwargs):
return render(request, 'index.html')
3、直接添加在類上,后面的name表示只給get添加裝飾器
from django.utils.decorators import method_decorator
@method_decorator(login_test, name='get')
class IndexView(View):
注意:
添加裝飾器前必須導入from django.utils.decorators import method_decorator
添加裝飾器的格式必須為@method_decorator(),括號里面為裝飾器的函數名
給類添加是必須聲明name
注意csrf-token裝飾器的特殊性,它只能加在dispatch上面
參考資料:
https://www.cnblogs.com/yuanchenqi/articles/8715364.html
https://www.jianshu.com/p/a3e6217d7bee