Django的CBV与FBV
FBV
FBV(function base views)?就是在視圖里使用函數處理請求。
在之前django的學習中,我們一直使用的是這種方式,所以不再贅述。
CBV
CBV(class base views)?就是在視圖里使用類處理請求。
Python是一個面向對象的編程語言,如果只用函數來開發,有很多面向對象的優點就錯失了(繼承、封裝、多態)。所以Django在后來加入了Class-Based-View。可以讓我們用類寫View。這樣做的優點主要下面兩種:
使用class-based views
如果我們要寫一個處理GET方法的view,用函數寫的話是下面這樣。
from django.http import HttpResponsedef my_view(request):if request.method == 'GET':return HttpResponse('OK')如果用class-based view寫的話,就是下面這樣
from django.http import HttpResponse from django.views import Viewclass 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 MyViewurlpatterns = [url(r'^index/$', MyView.as_view()), ]類的屬性可以通過兩種方法設置,第一種是常見的Python的方法,可以被子類覆蓋。
from django.http import HttpResponse from django.views import Viewclass GreetingView(View):name = "yuan"def get(self, request):return HttpResponse(self.name)# You can override that in a subclassclass 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寫在左邊。
回顧多重繼承和Mixin
多重繼承
回憶一下Animal類層次的設計,假設我們要實現以下4種動物:
- Dog - 狗狗;
- Bat - 蝙蝠;
- Parrot - 鸚鵡;
- Ostrich - 鴕鳥。
如果按照哺乳動物和鳥類歸類,我們可以設計出這樣的類的層次:
? ? ? ? ? ??
但是如果按照“能跑”和“能飛”來歸類,我們就應該設計出這樣的類的層次:
? ? ? ? ? ??
如果要把上面的兩種分類都包含進來,我們就得設計更多的層次:
- 哺乳類:能跑的哺乳類,能飛的哺乳類;
- 鳥類:能跑的鳥類,能飛的鳥類。
這么一來,類的層次就復雜了:
? ? ? ? ??
如果要再增加“寵物類”和“非寵物類”,這么搞下去,類的數量會呈指數增長,很明顯這樣設計是不行的。
正確的做法是采用多重繼承。首先,主要的類層次仍按照哺乳類和鳥類設計:
class Animal(object):pass# 大類: class Mammal(Animal):passclass Bird(Animal):pass# 各種動物: class Dog(Mammal):passclass Bat(Mammal):passclass Parrot(Bird):passclass Ostrich(Bird):pass現在,我們要給動物再加上Runnable和Flyable的功能,只需要先定義好Runnable和Flyable的類:
class Runnable(object):def run(self):print('Running...')class Flyable(object):def fly(self):print('Flying...'對于需要Runnable功能的動物,就多繼承一個Runnable,例如Dog:
class Dog(Mammal, Runnable):pass對于需要Flyable功能的動物,就多繼承一個Flyable,例如Bat:
class Bat(Mammal, Flyable):pass通過多重繼承,一個子類就可以同時獲得多個父類的所有功能。
Mixin
在設計類的繼承關系時,通常,主線都是單一繼承下來的,例如,Ostrich繼承自Bird。但是,如果需要“混入”額外的功能,通過多重繼承就可以實現,比如,讓Ostrich除了繼承自Bird外,再同時繼承Runnable。這種設計通常稱之為Mixin。
為了更好地看出繼承關系,我們把Runnable和Flyable改為RunnableMixin和FlyableMixin。類似的,你還可以定義出肉食動物CarnivorousMixin和植食動物HerbivoresMixin,讓某個動物同時擁有好幾個Mixin:
class Dog(Mammal, RunnableMixin, CarnivorousMixin):passMixin的目的就是給一個類增加多個功能,這樣,在設計類的時候,我們優先考慮通過多重繼承來組合多個Mixin的功能,而不是設計多層次的復雜的繼承關系。
Python自帶的很多庫也使用了Mixin。舉個例子,Python自帶了TCPServer和UDPServer這兩類網絡服務,而要同時服務多個用戶就必須使用多進程或多線程模型,這兩種模型由ForkingMixin和ThreadingMixin提供。通過組合,我們就可以創造出合適的服務來。
比如,編寫一個多進程模式的TCP服務,定義如下:
class MyTCPServer(TCPServer, ForkingMixin):pass編寫一個多線程模式的UDP服務,定義如下:
class MyUDPServer(UDPServer, ThreadingMixin):pass如果你打算搞一個更先進的協程模型,可以編寫一個CoroutineMixin:
class MyTCPServer(TCPServer, CoroutineMixin):pass這樣一來,我們不需要復雜而龐大的繼承鏈,只要選擇組合不同的類的功能,就可以快速構造出所需的子類。
小結
由于Python允許使用多重繼承,因此,Mixin就是一種常見的設計。只允許單一繼承的語言(如Java)不能使用Mixin的設計。
轉載于:https://www.cnblogs.com/gpd-Amos/p/8745022.html
總結
以上是生活随笔為你收集整理的Django的CBV与FBV的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 软件工程结对作业 四则运算界面设计
- 下一篇: Pycharm软件注册方法