理解Django 中Call Stack 機制的小Demo
- 2020 年 8 月 31 日
- 筆記
- Web development
1.工作流程
request/response模式下,request並不是直接到達view方法,view方法也不是將返回的response直接發送給瀏覽器的,而是request由外到里的層層通過各種middleware層,這個時候可以對request做一些事情,到最後一層也就是最內層時,得到view方法返回的response,然後再把這個response再由內到外層層傳遞出來,這時候可以對response做一些事情,如下圖:
2.原理
class SimpleMiddleware: def __init__(self, get_response): self.get_response = get_response # One-time configuration and initialization. def __call__(self, request): # Code to be executed for each request before # the view (and later middleware) are called. response = self.get_response(request) # Code to be executed for each request/response after # the view is called. return response
每個middleware都如上面代碼一樣,有兩個必須的函數:(1)__init__(self,get_response),負責在web server啟動時完成初始化,即建立與下一層middleware的聯繫.(2)__call__(self,request),在每次request下,被調用。
有三個可選的函數:(1)process_view:__init__和__call__對view 方法一無所知,而process_view給了通道(give access to)在調用view方法之前獲曉view方法及其參數,如果出現了,則它在__call__調用後,在self.get_response(request)之前調用,因此可以觸發view 方法。(2)process_exception:如果在view方法中出現異常,該函數可以提供機會來處理異常(3)process_template_response:在self.get_response(request)調用後,view方法結束後,提供修改response的機會,需要注意的是該函數僅僅在view方法返回的是TemplateResponse類的情況下有效,如果返回的是render() ,則該函數不被觸發。
詳見//docs.djangoproject.com/en/3.1/topics/http/middleware/
(3)供進一步理解call stack 機制的小demo(僅供參考)
import time class Base: def __init__(self,get_response): self.get_response=get_response def __call__(self,request=None): self.pre() response=self.get_response(request) self.after() return response def pre(self): pass def after(self): pass class Response: def __init__(self): pass def view(request): return Response() def startServer(callstackSets,view): callstackSets.reverse() last=callstackSets.pop(0)(view) for i in range(len(callstackSets)): last=callstackSets.pop(0)(last) return last class A(Base): def pre(self): print('In A') def after(self): print('Out A ') class B(Base): def pre(self): print('In B') def after(self): print('Out B ') class C(Base): def pre(self): print('In C') def after(self): print('Out C ') if __name__=='__main__': callstackSets=[A,B,C] calls=startServer(callstackSets,view) calls('req')
運行後:
可以看到完成了既定的目標,request進來後,一層一層的進入,直到最內層C,調用view方法,得到response,然後將response從最內層一層一層的傳遞出來!