Django中间件简析

上一篇简单的叙述了CSRF这个中间件的作用,他在执行视图函数之前可以对csrftoken进行验证,如果通过才执行否则直接报错。


那么什么是中间件呢?这个要回到Django的生命周期里面。一个基本的生命周期是用户输入一个URL,通过urls.py找到对应的视图函数,然后进行数据处理,返回渲染后的结果。在url和视图函数的匹配过程中,还有一个重要的过程,就是依次执行所有的中间件的类里面的函数。


还是以csrf中间件为例,看看他的源码截图如下,他继承了一个MiddlewareMixin 类,里面还定义了一些特殊的固定名字的函数,例如 process_view, process_response等等,如果需要自定义一个中间件,我们也需要这些东西。

Django中间件简析


下面看看实例来解释他们的用途。


定义一个m1.py文件,里面是我们自定义的中间件,这里创建了3个类,每个类就是一个中间件

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
rom django.utils.deprecation import MiddlewareMixin
class 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

下面是系统默认的中间件配置,他的执行顺序是从上往下执行的。如果需要添加自定义的中间件,可以直接添加在下面

1
2
3
4
5
6
7
8
9
10
11
12
13
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',
    
]


最后是我的视图函数

1
2
3
def test(request):
    print('执行View函数')
    return HttpResponse('ok')


整个流程的执行顺序是发送Url请求,然后中间件按从上到下顺序执行自己的 process_request函数,然后掉过头来再从上到下执行process_view函数,然后到达视图函数,如果有错误,按照从下往上的顺序来执行 process_exception函数,如果无误,执行每个中间件的process_response函数


下图转自网络


Django中间件简析


因此如果输入http://127.0.0.1:8000/test  ,控制台输出结果是

1
2
3
4
5
6
7
8
9
10
R11
R21
R31
R12
R22
R32
执行View函数
R33
R23
R13


如果在process_view里面执行了HttpResponse的返回操作,那么他会直接跳过process_view和视图函数,而直接跳到当前中间件的process_response,然后一路返回


例如:修改Row2

1
2
3
4
5
from django.shortcuts import HttpResponse
class Row2(MiddlewareMixin):
    def process_request(self,request):
        print('R21')
        return HttpResponse('走')

那么结果直接显示


Django中间件简析

1
2
3
4
R11
R21
R23
R13


如果我故意执行一个报错的代码,比如

1
2
3
4
def test(request):
    print('执行View函数')
    int('sdsfsdfs')
    return HttpResponse('ok')


Django中间件简析






本文转自 beanxyz 51CTO博客,原文链接:http://blog.51cto.com/beanxyz/1963054,如需转载请自行联系原作者