理解Django 中Call Stack 机制的小Demo
1.工作流程
request/response模式下,request并不是直接到达view方法,view方法也不是将返回的response直接发送给浏览器的,而是request由外到里的层层通过各种middleware层,这个时候可以对request做一些事情,到最后一层也就是最内层时,得到view方法返回的response,然后再把这个response再由内到外层层传递出来,这时候可以对response做一些事情,如下图:
2.原理
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | 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(仅供参考)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 | 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从最内层一层一层的传递出来!
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· 【自荐】一款简洁、开源的在线白板工具 Drawnix