django中視圖處理請求方式(FBV、CBV)


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和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


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM