理解Django 中Call Stack 機制的小Demo

JohnYang819發表於2020-08-31

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() ,則該函式不被觸發。

詳見https://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從最內層一層一層的傳遞出來!

 

相關文章