Django學(xué)習(xí)筆記之Class-Based-View
前言
大家都知道其實(shí)學(xué)習(xí)Django非常簡(jiǎn)單,幾乎不用花什么精力就可以入門(mén)了。配置一個(gè)url,分給一個(gè)函數(shù)處理它,返回response,幾乎都沒(méi)有什么很難理解的地方。
寫(xiě)多了,有些問(wèn)題才逐漸認(rèn)識(shí)到。比如有一個(gè)view比較復(fù)雜,調(diào)用了很多其他的函數(shù)。想要把這些函數(shù)封裝起來(lái),怎么辦?當(dāng)然,可以用注釋#------view------這樣將函數(shù)隔離開(kāi),這種方法太low了,簡(jiǎn)直是在騙自己,連封裝都算不上。
Python是一個(gè)面向?qū)ο蟮木幊陶Z(yǔ)言,如果只用函數(shù)來(lái)開(kāi)發(fā),有很多面向?qū)ο蟮膬?yōu)點(diǎn)就錯(cuò)失了(繼承、封裝、多態(tài))。所以Django在后來(lái)加入了Class-Based-View??梢宰屛覀冇妙?lèi)寫(xiě)View。這樣做的優(yōu)點(diǎn)主要下面兩種:
- 提高了代碼的復(fù)用性,可以使用面向?qū)ο蟮募夹g(shù),比如Mixin(多繼承)
- 可以用不同的函數(shù)針對(duì)不同的HTTP方法處理,而不是通過(guò)很多if判斷,提高代碼可讀性
使用class-based views
如果我們要寫(xiě)一個(gè)處理GET方法的view,用函數(shù)寫(xiě)的話是下面這樣。
from django.http import HttpResponse def my_view(request): if request.method == 'GET': # <view logic> return HttpResponse('result')
如果用class-based view寫(xiě)的話,就是下面這樣。
from django.http import HttpResponse from django.views import View class MyView(View): def get(self, request): # <view logic> return HttpResponse('result')
Django的url是將一個(gè)請(qǐng)求分配給可調(diào)用的函數(shù)的,而不是一個(gè)class。針對(duì)這個(gè)問(wèn)題,class-based view提供了一個(gè)as_view()
靜態(tài)方法(也就是類(lèi)方法),調(diào)用這個(gè)方法,會(huì)創(chuàng)建一個(gè)類(lèi)的實(shí)例,然后通過(guò)實(shí)例調(diào)用dispatch()
方法,dispatch()
方法會(huì)根據(jù)request的method的不同調(diào)用相應(yīng)的方法來(lái)處理request(如get()
, post()
等)。到這里,這些方法和function-based view差不多了,要接收request,得到一個(gè)response返回。如果方法沒(méi)有定義,會(huì)拋出HttpResponseNotAllowed異常。
在url中,就這么寫(xiě):
# urls.py from django.conf.urls import url from myapp.views import MyView urlpatterns = [ url(r'^about/$', MyView.as_view()), ]
類(lèi)的屬性可以通過(guò)兩種方法設(shè)置,第一種是常見(jiàn)的Python的方法,可以被子類(lèi)覆蓋。
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中指定類(lèi)的屬性:
在url中設(shè)置類(lèi)的屬性Python
urlpatterns = [ url(r'^about/$', GreetingView.as_view(greeting="G'day")), ]
使用Mixin
我覺(jué)得要理解django的class-based-view(以下簡(jiǎn)稱(chēng)cbv),首先要明白django引入cbv的目的是什么。在django1.3之前,generic view也就是所謂的通用視圖,使用的是function-based-view(fbv),亦即基于函數(shù)的視圖。有人認(rèn)為fbv比cbv更pythonic,竊以為不然。python的一大重要的特性就是面向?qū)ο蟆6鴆bv更能體現(xiàn)python的面向?qū)ο?。cbv是通過(guò)class的方式來(lái)實(shí)現(xiàn)視圖方法的。class相對(duì)于function,更能利用多態(tài)的特定,因此更容易從宏觀層面上將項(xiàng)目?jī)?nèi)的比較通用的功能抽象出來(lái)。關(guān)于多態(tài),不多解釋?zhuān)信d趣的同學(xué)自己Google??傊梢岳斫鉃橐粋€(gè)東西具有多種形態(tài)(的特性)。cbv的實(shí)現(xiàn)原理通過(guò)看django的源碼就很容易明白,大體就是由url路由到這個(gè)cbv之后,通過(guò)cbv內(nèi)部的dispatch方法進(jìn)行分發(fā),將get請(qǐng)求分發(fā)給cbv.get方法處理,將post請(qǐng)求分發(fā)給cbv.post方法處理,其他方法類(lèi)似。怎么利用多態(tài)呢?cbv里引入了mixin的概念。Mixin就是寫(xiě)好了的一些基礎(chǔ)類(lèi),然后通過(guò)不同的Mixin組合成為最終想要的類(lèi)。
所以,理解cbv的基礎(chǔ)是,理解Mixin。Django中使用Mixin來(lái)重用代碼,一個(gè)View Class可以繼承多個(gè)Mixin,但是只能繼承一個(gè)View(包括View的子類(lèi)),推薦把View寫(xiě)在最右邊,多個(gè)Mixin寫(xiě)在左邊。Mixin也是比較復(fù)雜的技術(shù),本文不詳細(xì)說(shuō)了,以后寫(xiě)一篇針對(duì)Mixin的文章吧。
使用裝飾器
在CBV中,可以使用method_decorator來(lái)裝飾方法。
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)
也可以寫(xiě)在類(lèi)上面,傳入方法的名字。
@method_decorator(login_required, name='dispatch') class ProtectedView(TemplateView): template_name = 'secret.html'
如果有多個(gè)裝飾器裝飾一個(gè)方法,可以寫(xiě)成一個(gè)list。例如,下面這兩種寫(xiě)法是等價(jià)的。
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'
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問(wèn)大家可以留言交流。
相關(guān)文章
關(guān)于python scrapy中添加cookie踩坑記錄
這篇文章主要介紹了關(guān)于python scrapy中添加cookie踩坑記錄,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11遠(yuǎn)程部署工具Fabric詳解(支持Python3)
這篇文章主要介紹了遠(yuǎn)程部署神器 Fabric,支持Python3 ,Fabric 出場(chǎng)了。Fabric 是一個(gè)遠(yuǎn)程部署神器,它可以在本地執(zhí)行遠(yuǎn)程服務(wù)器的命令。,需要的朋友可以參考下2019-07-07對(duì)python中l(wèi)ist的拷貝與numpy的array的拷貝詳解
今天小編就為大家分享一篇對(duì)python中l(wèi)ist的拷貝與numpy的array的拷貝詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-01-01Pytorch四維Tensor轉(zhuǎn)圖片并保存方式(維度順序調(diào)整)
這篇文章主要介紹了Pytorch四維Tensor轉(zhuǎn)圖片并保存方式(維度順序調(diào)整),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12Flask實(shí)現(xiàn)的接口響應(yīng)中存在中文時(shí)接口返回為unicode亂碼的解決方法
本文給大家分享了新版Flask實(shí)現(xiàn)的接口響應(yīng)中存在中文時(shí)接口返回為unicode亂碼的解決方法,文中通過(guò)代碼示例和圖文介紹的非常詳細(xì),如果有遇到相同問(wèn)題的朋友,可以參考閱讀本文2023-11-11Python logging管理不同級(jí)別log打印和存儲(chǔ)實(shí)例
這篇文章主要介紹了Python logging管理不同級(jí)別log打印和存儲(chǔ)實(shí)例,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01Python使用擴(kuò)展庫(kù)pywin32實(shí)現(xiàn)批量文檔打印實(shí)例
這篇文章主要介紹了Python使用擴(kuò)展庫(kù)pywin32實(shí)現(xiàn)批量文檔打印實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-04-04