上一篇简单的叙述了CSRF这个中间件的作用,他在执行视图函数之前可以对csrftoken进行验证,如果通过才执行否则直接报错。
那么什么是中间件呢?这个要回到Django的生命周期里面。一个基本的生命周期是用户输入一个URL,通过urls.py找到对应的视图函数,然后进行数据处理,返回渲染后的结果。在url和视图函数的匹配过程中,还有一个重要的过程,就是依次执行所有的中间件的类里面的函数。
还是以csrf中间件为例,看看他的源码截图如下,他继承了一个MiddlewareMixin 类,里面还定义了一些特殊的固定名字的函数,例如 process_view, process_response等等,如果需要自定义一个中间件,我们也需要这些东西。
下面看看实例来解释他们的用途。
定义一个m1.py文件,里面是我们自定义的中间件,这里创建了3个类,每个类就是一个中间件
rom django.utils.deprecation import MiddlewareMixinclass Row1(MiddlewareMixin):
def process_request(self,request):
print('R11')
def process_view(self, request, view_func, view_func_args, view_func_kwargs):
print('R12')
def process_response(self, request, response):
print('R13')
return response
from django.shortcuts import HttpResponse
class Row2(MiddlewareMixin):
def process_request(self,request):
print('R21')
# return HttpResponse('走')
def process_view(self, request, view_func, view_func_args, view_func_kwargs):
print('R22')
def process_response(self, request, response):
print('R23')
return response
class Row3(MiddlewareMixin):
def process_request(self,request):
print('R31')
def process_view(self, request, view_func, view_func_args, view_func_kwargs):
print('R32')
def process_response(self, request, response):
print('R33')
return response
def process_exception(self, request, exception):
if isinstance(exception,ValueError):
return HttpResponse('出现异常》。。')
def process_template_response(self,request,response):
# 如果Views中的函数返回的对象中,具有render方法
print('-----------------------')
return response
下面是系统默认的中间件配置,他的执行顺序是从上往下执行的。如果需要添加自定义的中间件,可以直接添加在下面
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', 'Middle.m1.Row1', 'Middle.m1.Row2', 'Middle.m1.Row3', ]
最后是我的视图函数
def test(request): print('执行View函数') return HttpResponse('ok')
整个流程的执行顺序是发送Url请求,然后中间件按从上到下顺序执行自己的 process_request函数,然后掉过头来再从上到下执行process_view函数,然后到达视图函数,如果有错误,按照从下往上的顺序来执行 process_exception函数,如果无误,执行每个中间件的process_response函数
下图转自网络
因此如果输入http://127.0.0.1:8000/test ,控制台输出结果是
R11R21R31R12R22R32执行View函数R33R23R13
如果在process_view里面执行了HttpResponse的返回操作,那么他会直接跳过process_view和视图函数,而直接跳到当前中间件的process_response,然后一路返回
例如:修改Row2
from django.shortcuts import HttpResponseclass Row2(MiddlewareMixin): def process_request(self,request): print('R21') return HttpResponse('走')
那么结果直接显示
R11R21R23R13
如果我故意执行一个报错的代码,比如
def test(request): print('执行View函数') int('sdsfsdfs') return HttpResponse('ok')
本文出自 “麻婆豆腐” 博客,请务必保留此出处http://beanxyz.blog.51cto.com/5570417/1963054