詳解Django中間件執(zhí)行順序
中間件
Django中的中間件是一個輕量級、底層的插件系統(tǒng),可以介入Django的請求和響應處理過程,修改Django的輸入或輸出。中間件的設計為開發(fā)者提供了一種無侵入式的開發(fā)方式,增強了Django框架的健壯性。
我們可以使用中間件,在Django處理視圖的不同階段對輸入或輸出進行干預。
1 中間件的定義方法
定義一個中間件工廠函數(shù),然后返回一個可以別調(diào)用的中間件。
中間件工廠函數(shù)需要接收一個可以調(diào)用的get_response對象。
返回的中間件也是一個可以被調(diào)用的對象,并且像視圖一樣需要接收一個request對象參數(shù),返回一個response對象。
def simple_middleware(get_response): # 此處編寫的代碼僅在Django第一次配置和初始化的時候執(zhí)行一次。 def middleware(request): # 此處編寫的代碼會在每個請求處理視圖前被調(diào)用。 response = get_response(request) # 此處編寫的代碼會在每個請求處理視圖之后被調(diào)用。 return response return middleware
例如,在users應用中新建一個middleware.py文件,
def my_middleware(get_response): print('init 被調(diào)用') def middleware(request): print('before request 被調(diào)用') response = get_response(request) print('after response 被調(diào)用') return response return middleware
定義好中間件后,需要在settings.py 文件中添加注冊中間件
MIDDLEWARE = [ 'django.middleware.security.SecurityMiddleware', 'django.contrib.sessions.middleware.SessionMiddleware', 'django.middleware.common.CommonMiddleware', # 'django.middleware.csrf.CsrfViewMiddleware', 'django.contrib.auth.middleware.AuthenticationMiddleware', 'django.contrib.messages.middleware.MessageMiddleware', 'django.middleware.clickjacking.XFrameOptionsMiddleware', 'users.middleware.my_middleware', # 添加中間件 ]
定義一個視圖進行測試
def demo_view(request): print('view 視圖被調(diào)用') return HttpResponse('OK')
執(zhí)行結(jié)果
注意:Django運行在調(diào)試模式下,中間件init部分有可能被調(diào)用兩次。
2 執(zhí)行流程
3 多個中間件的執(zhí)行順序
- 在請求視圖被處理前,中間件由上至下依次執(zhí)行
- 在請求視圖被處理后,中間件由下至上依次執(zhí)行
示例:
定義兩個中間件
def my_middleware(get_response): print('init 被調(diào)用') def middleware(request): print('before request 被調(diào)用') response = get_response(request) print('after response 被調(diào)用') return response return middleware def my_middleware2(get_response): print('init2 被調(diào)用') def middleware(request): print('before request 2 被調(diào)用') response = get_response(request) print('after response 2 被調(diào)用') return response return middleware
注冊添加兩個中間件
MIDDLEWARE = [ 'django.middleware.security.SecurityMiddleware', 'django.contrib.sessions.middleware.SessionMiddleware', 'django.middleware.common.CommonMiddleware', # 'django.middleware.csrf.CsrfViewMiddleware', 'django.contrib.auth.middleware.AuthenticationMiddleware', 'django.contrib.messages.middleware.MessageMiddleware', 'django.middleware.clickjacking.XFrameOptionsMiddleware', 'users.middleware.my_middleware', # 添加 'users.middleware.my_middleware2', # 添加 ]
執(zhí)行結(jié)果
init2 被調(diào)用
init 被調(diào)用
before request 被調(diào)用
before request 2 被調(diào)用
view 視圖被調(diào)用
after response 2 被調(diào)用
after response 被調(diào)用
總結(jié):中間件的本質(zhì)其實就是個裝飾器,對于裝飾器我之前的隨筆里也詳細說過,裝飾器的本質(zhì)其實就是個閉包
對于裝飾器來說,在這里程序從上到下執(zhí)行,開始記錄裝飾器1-3,然后讀到了函數(shù)的時候,裝飾器開始裝飾,把函數(shù)的引用傳入裝飾器中,從裝飾器3開始往上裝飾,所以這時候開始執(zhí)行裝飾器3的初始化,并把裝飾完的函數(shù)當做一個新的函數(shù),再次把新的引用傳入到裝飾器2,接著裝飾器2進行初始化,再次把新的函數(shù)的引用傳入到裝飾器1進行裝飾,這時候裝飾器1的初始化開始,并開始執(zhí)行,從而接下來的執(zhí)行順序為1-3執(zhí)行裝飾的內(nèi)容,最后再執(zhí)行本來的函數(shù),達到一個對原有函數(shù)增加功能和特性的要求。
裝飾器:從程序開始的順序,從上到下讀取----》從下到上裝飾----》從上到下執(zhí)行
在中間件中,執(zhí)行順序等同,只不過中間件裝飾的是視圖函數(shù)或者是試圖類,會有一個init()初始化,為了便于理解我在說明裝飾器順序的時候,那里說的初始化就是現(xiàn)在的init(),也就是中間件在執(zhí)行上也是從最接近函數(shù)或者類的中間件開始初始化,在中間件注冊時,如果有多個中間件,程序按照從上到下的順序來確認中間件,接著執(zhí)行視圖函數(shù)的時候,開始從下到上的進行init(),這里的初始化init()類似于Flask框架里四種請求勾子中的請求開始前執(zhí)行一次的意思,從下到上執(zhí)行完init后,再從上到下執(zhí)行請求前所要執(zhí)行的代碼,請求完成后再從下往上執(zhí)行請求后的代碼。
中間件:從程序開始的順序,從上到下讀取----》從下到上初始化init()----》從上到下執(zhí)行請求前----》視圖的請求----》從下到上執(zhí)行請求后
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
淺析如何在Python中使用結(jié)構(gòu)模式匹配
在Python 3.10中引入了模式匹配語法,允許我們在應用程序中使用強大的新編程技術進行決策,下面我們就來講講如何在Python中使用結(jié)構(gòu)模式匹配吧2023-08-08django的聚合函數(shù)和aggregate、annotate方法使用詳解
這篇文章主要介紹了django的聚合函數(shù)和aggregate、annotate方法使用詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2019-07-07