• 08.django之中间件


    一、中间件介绍

    # 中间件简单来说,就是给所有的请求都加上了相同的操作或功能
    # 如果想修改传到视图的HTTPRequest对象,或修改view返回的HttpResponse对象,都可以通过中间件实现
    """
    说白了,就是帮助我们在视图函数执行之前和之后做一些额外操作.
    本质就是一个类,类中有几个方法,django会自动执行相应的方法
    """

    二、django请求生命周期

    • 当用户在浏览器中输入url时,浏览器会生成请求头和请求体发给服务端,请求头和请求体中会包含浏览器的动作(action),这个动作通常为get或者post
    • url经过Django中的wsgi,wsgi中封装了socket,然后按照http协议解包
    • 接着会经过中间件,按顺序执行每个中间件的request方法
    • 最后经过url路由映射表,在路由系统中一条一条进行匹配,一旦其中一条匹配成功就执行对应的视图函数,后面的路由就不再继续匹配了.
    • 视图函数根据客户端的请求查询相应的数据,经过中间件按倒序执行每个中间件的response方法
    • 接着通过wsgi按http协议封装相应数据
    • 最后Django把客户端想要的数据做为一个字符串返回给客户端,客户端浏览器接收到返回的数据,经过渲染后显示给用户.

    三、中间件方法及自定义中间件

    1、中间件方法

    • 中间件五种方法,分别为:
    • process_request(self,request)
    • process_response(self, request, response)
    • process_exception(self, request, exception)
    • process_view(self, request, view_func, view_args, view_kwargs)
    • process_template_response(self,request,response)

      以上方法的返回值可以是None或一个HttpResponse对象,如果是None,则继续按照django定义的规则向后继续执行,如果是HttpResponse对象,则直接将该对象返回给用户。

      当用户发起请求的时候会依次经过所有的的中间件,这个时候的请求时process_request,最后到达views的函数中,views函数处理后,在依次穿过中间件,这个时候是process_response,最后返回给请求者。

    2、自定义中间件

    # 在项目中创建一个包,随便起名字,一般都放在一个叫做utils的包里面,表示一个公用的组件,
    # 创建一个py文件,随便起名字,例如叫做:middlewares.py,内容如下
    from django.utils.deprecation import MiddlewareMixin
    
    class MD1(MiddlewareMixin):
        #自定义中间件,不是必须要有下面这两个方法,有request方法说明请求来了要处理,有response方法说明响应出去时需要处理,
        #不是非要写这两个方法,如果你没写process_response方法,那么会一层一层的往上找,哪个中间件有process_response方法就将返回对象给哪个中间件
        def process_request(self, request):
            print("MD1里面的 process_request")
    
        def process_response(self, request, response):
            print("MD1里面的 process_response")
            return response

    3、关于中间件方法详细

    3.1、process_request

    process_request有一个参数,就是request,这个request和视图函数中的request是一样的

    它的返回值可以是None也可以是HttpResponse对象。返回值是None的话,按正常流程继续走,交给下一个中间件处理,如果是HttpResponse对象,Django将不执行视图函数,而将相应对象返回给浏览器。

    多个中间件都有process_request怎么执行的

    from django.utils.deprecation import MiddlewareMixin
    
    
    class MD1(MiddlewareMixin):
    
        def process_request(self, request):
            print("MD1里面的 process_request")
    
    
    class MD2(MiddlewareMixin):
        def process_request(self, request):
            print("MD2里面的 process_request")
            pass

    在settings.py的MIDDLEWARE配置项中注册两个自定义中间件:

    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',
        'middlewares.MD1',  
        # 自定义中间件MD1,这个写的是你项目路径下的一个路径,
        # 例如,如果你放在项目下,文件夹名成为utils,那么这里应该写utils.middlewares.MD1
        'middlewares.MD2'  # 自定义中间件MD2
    ]

    此时,访问一个视图,会自动在MIDDLEWARE列表从上到下执行,发现终端打印如下内容:

    MD1里面的 process_request
    MD2里面的 process_request
    app01 中的 index视图

    把settings里MIDDLEWARE中MD1和MD2调换一下:

    MD2里面的 process_request
    MD1里面的 process_request
    app01 中的 index视图

    哪个中间件在前先执行哪个,视图函数是最后执行的,MD2比MD1先执行自己的process_request方法。

    总结:

    """
    中间件的process_request方法是在执行视图函数之前执行的。
    当配置多个中间件时,会按照MIDDLEWARE中的注册顺序,也就是列表的索引值,从前到后依次执行的。
    不同中间件之间传递的request都是同一个对象
    """

    3.2、process_response

    多个中间件中的process_response方法是按照MIDDLEWARE中的注册顺序倒序执行的,也就是说第一个中间件的process_request方法首先执行,而它的process_response方法最后执行,最后一个中间件的process_request方法最后一个执行,它的process_response方法是最先执行。

    """
    它有两个参数,一个是request,一个是response,request就是上述例子中一样的对象,
    response是视图函数返回的HttpResponse对象。该方法的返回值也必须是HttpResponse对象。
    """

    给上面M1和M2加上process_response方法

    from django.utils.deprecation import MiddlewareMixin
    
    
    class MD1(MiddlewareMixin):
    
        def process_request(self, request):
            print("MD1里面的 process_request")
            #不必须写return值
        def process_response(self, request, response):#request和response两个参数必须有,名字随便取
            print("MD1里面的 process_response")
            #print(response.__dict__['_container'][0].decode('utf-8')) #查看响应体里面的内容的方法,或者直接使用response.content也可以看到响应体里面的内容,
    # 由于response是个变量,直接点击看源码是看不到的,你打印type(response)发现是HttpResponse对象,
    # 查看这个对象的源码就知道有什么方法可以用了。
         return response  #必须有返回值,写return response  ,这个response就像一个接力棒一样
            #return HttpResponse('瞎搞') ,如果你写了这个,那么你视图返回过来的内容就被它给替代了
    
    class MD2(MiddlewareMixin):
        def process_request(self, request):
            print("MD2里面的 process_request")
            pass
    
        def process_response(self, request, response): #request和response两个参数必须要有,名字随便取
            print("MD2里面的 process_response") 
            return response  #必须返回response,不然你上层的中间件就没有拿到httpresponse对象,就会报错

    访问视图,查看结果:

    MD2里面的 process_request
    MD1里面的 process_request
    app01 中的 index视图
    MD1里面的 process_response
    MD2里面的 process_response

    settings.py中MD2比MD1先注册,process_response方法是在视图函数之后执行的,并且顺序是MD1比MD2先执行

    多个中间件中的process_response方法是按照MIDDLEWARE中的注册顺序倒序执行的,也就是说第一个中间件的process_request方法首先执行,而它的process_response方法最后执行,最后一个中间件的process_request方法最后一个执行,它的process_response方法是最先执行。

    如果当请求到达请求2的时候直接不符合条件返回,即return HttpResponse("Md2中断"),程序将把请求直接发给中间件2返回,然后依次返回到请求者,结果如下:

    MD2里面的 process_request
    MD1里面的 process_request
    MD1里面的 process_response
    MD2里面的 process_response

    流程如下:

    例:

    基于session的登录验证:

    # 基于session的登录认证中间件
    class LoginAuth(MiddlewareMixin):
        
        white_list = ['/login/', ]  # 白名单
        def process_request(self,request):
            path = request.path
            #设置路径白名单,只要访问的是login登陆路径,就不做这个cookie认证
            if path not in self.white_list:
                is_login = request.session.get('is_login')
                print(is_login, type(is_login)) #True <class 'bool'>
    
                # request.session['is_login']
                # 1 取出请求中cookie键为sessionid的值
                # 2 通过这个值到django-session表中获取数据
                # 3 将数据解密并且反序列化得到原来的数据
    
                # is_login = request.COOKIES.get('is_login')
                if is_login != True:
                    return redirect('/login/')
    from django.conf.urls import url
    from django.contrib import admin
    from app01 import views
    urlpatterns = [
        url(r'^admin/', admin.site.urls),
        url(r'^login/', views.login),
        url(r'^home/', views.home)
    ]
    urls.py
    def login(request):
        if request.method == 'GET':
            return render(request, 'login.html')
        else:
            uname = request.POST.get('username')
            if uname == 'root' or uname=='abc':
                request.session['is_login'] = True
                request.session['username'] = uname
    
                # request.session
                #1 生成一个随机字符串
                #2 将随机字符串放到cookie中,名称为sessionid
                #3 将设置的session数据,序列化+加密,保存到了django-session表中
    
                return redirect('/home/')
            else:
                return redirect('/login/')
    views.py
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    <form action="" method="post">
        用户名: <input type="text" name="username">
        密码: <input type="password" name="password">
        <input type="submit">
    </form>
    
    
    </body>
    </html>
    login.html

    3.3、process_view

    process_view(self, request, view_func, view_args, view_kwargs)

          该方法有四个参数

          request是HttpRequest对象。

          view_func是Django即将使用的视图函数。 (它是实际的函数对象,而不是函数的名称作为字符串。)

          view_args是将传递给视图的位置参数的列表.

          view_kwargs是将传递给视图的关键字参数的字典。 view_args和view_kwargs都不包含第一个视图参数(request)。

          Django会在调用视图函数之前调用process_view方法。

          它应该返回None或一个HttpResponse对象。 如果返回None,Django将继续处理这个请求,执行任何其他中间件的process_view方法,然后在执行相应的视图。 如果它返回一个HttpResponse对象,Django不会调用对应的视图函数。 它将执行中间件的process_response方法并将应用到该HttpResponse并返回结果。

    给MD1和MD2添加process_view方法:

    from django.utils.deprecation import MiddlewareMixin
    
    
    class MD1(MiddlewareMixin):
    
        def process_request(self, request):
            print("MD1里面的 process_request")
    
        def process_response(self, request, response):
            print("MD1里面的 process_response")
            return response
    
        def process_view(self, request, view_func, view_args, view_kwargs):
            print("-" * 80)
            print("MD1 中的process_view")
            print(view_func, view_func.__name__) #就是url映射到的那个视图函数,也就是说每个中间件的这个process_view已经提前拿到了要执行的那个视图函数
            #ret = view_func(request) #提前执行视图函数,不用到了上图的试图函数的位置再执行,如果你视图函数有参数的话,可以这么写 view_func(request,view_args,view_kwargs) 
            #return ret  #直接就在MD1中间件这里这个类的process_response给返回了,就不会去找到视图函数里面的这个函数去执行了。
    
    class MD2(MiddlewareMixin):
        def process_request(self, request):
            print("MD2里面的 process_request")
            pass
    
        def process_response(self, request, response):
            print("MD2里面的 process_response")
            return response
    
        def process_view(self, request, view_func, view_args, view_kwargs):
            print("-" * 80)
            print("MD2 中的process_view")
            print(view_func, view_func.__name__)

    访问视图,得到结果:

    MD2里面的 process_request
    MD1里面的 process_request
    --------------------------------------------------------------------------------
    MD2 中的process_view
    <function index at 0x000001DE68317488> index
    --------------------------------------------------------------------------------
    MD1 中的process_view
    <function index at 0x000001DE68317488> index
    app01 中的 index视图
    MD1里面的 process_response
    MD2里面的 process_response

                                                                                                                           

  • 相关阅读:
    旋转加载loading和点点加载loadingdemo
    css 点点加载demo
    gulp——myself配置
    AngularJS官网seed目录结构
    CSS content换行技术实现字符animation loading效果
    gulp入门与一些基本设置
    css 图标 旋转中
    【图文教程】WebStorm下使用Github下载以及上传代码
    gulp-uglify的使用
    面试题 ——— 二维数组的查找
  • 原文地址:https://www.cnblogs.com/kongxiangqun/p/13714422.html
Copyright © 2020-2023  润新知