• django组件:中间件


    全局性的逻辑处理

    一、中间件的概念

    中间件顾名思义,是介于request与response处理之间的一道处理过程,相对比较轻量级,并且在全局上改变django的输入与输出。因为改变的是全局,所以需要谨慎使用,用不好会影响到性能。
    Django的中间件的定义:

    Middleware is a framework of hooks into Django’s request/response processing. <br>It’s a light, low-level “plugin” system for globally altering Django’s input or output.

    如果你想修改请求,例如被传送到view中的HttpRequest对象。 或者你想修改view返回的HttpResponse对象,这些都可以通过中间件来实现。
    可能你还想在view执行之前做一些操作,这种情况就可以用 middleware来实现。

    setting.py

    Django 默认的Middleware:默认定义7个

    一个中间件就是一个类

    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',
    ]

    中间件起到什么角色,一个完整django经历那些阶段

    自定义中间件

    中间件中一共有四个方法:
    
    process_request
    process_view
    process_exception
    process_response

    process_request,process_response

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

    上述截图中的中间件都是django中的,我们也可以自己定义一个中间件,我们可以自己写一个类,但是必须继承MiddlewareMixin
    需要导入

    from django.utils.deprecation import MiddlewareMixin

    无论访问的路径是什么,中间件都会执行

    url

    from django.contrib import admin
    from django.urls import path
    from app01 import views
    urlpatterns = [
        path('admin/', admin.site.urls),
        path('index/', views.index),
        path('index_new/', views.index_new),
    ]

    views.py

    from django.shortcuts import render,HttpResponse
    
    # Create your views here.
    def index(request):
        return HttpResponse('index')
    
    def index_new(request):
        return HttpResponse('is new index')

    自定义中间件

    my_middlewares.py

    from django.utils.deprecation import MiddlewareMixin
    from django.shortcuts import HttpResponse
    
    class CustomerMiddleware(MiddlewareMixin):
        def process_request(self,request):
            print("CustomerMiddleware1 process_request....")
            #return HttpResponse("forbidden....")
    
        def process_response(self,request,response):
            print("CustomerMiddleware1 process_response")
    
            return response
            #return HttpResponse("hello yuan")
        def process_view(self, request, callback, callback_args, callback_kwargs):
            print("CustomerMiddleware1 process_view")
    
        def process_exception(self, request, exception):
    
            print("CustomerMiddleware1 process_exception")
            return HttpResponse(exception)
    
    class CustomerMiddleware2(MiddlewareMixin):
    
        def process_request(self,request):
            print("CustomerMiddleware2 process_request....")
    
        def process_response(self,request,response):
            print("CustomerMiddleware2 process_response")
            return response
    
        def process_view(self, request, callback, callback_args, callback_kwargs):
            # print("====>",callback(callback_args))
            print("CustomerMiddleware2 process_view")
            # ret=callback(callback_args)
            # return ret
    
        def process_exception(self, request, exception):
    
            print("CustomerMiddleware2 process_exception")
    
            #return HttpResponse(exception)

    在settings配置

    分别访问index index_new

    中间件都会执行

    CustomerMiddleware1 process_request....
    CustomerMiddleware2 process_request....
    CustomerMiddleware1 process_view
    CustomerMiddleware2 process_view
    CustomerMiddleware2 process_response
    CustomerMiddleware1 process_response

     在view.py

    def index(request):
        print('-----index-----')
        return HttpResponse('index')

    在浏览器中访问index

    CustomerMiddleware1 process_request....
    CustomerMiddleware2 process_request....
    CustomerMiddleware1 process_view
    CustomerMiddleware2 process_view
    -----index-----  # view.py视图
    CustomerMiddleware2 process_response
    CustomerMiddleware1 process_response
    先请求--视图--响应

    具体分析:

    1、当自定义的中间件中只有请求和相应 ,:

     def process_request(self,request):
            print("CustomerMiddleware1 process_request....")
            #return HttpResponse("forbidden....")
    
        def process_response(self,request,response):
            print("CustomerMiddleware1 process_response")
    
    
    
    
    

     2、当自定义的中间件有process_view

    
    
    class CustomerMiddleware(MiddlewareMixin):

    def process_request(self,request):
    print("CustomerMiddleware1 process_request....")

    #return HttpResponse("forbidden....")


    def process_response(self,request,response):
    print("CustomerMiddleware1 process_response")

    return response
    #return HttpResponse("hello yuan")

    def process_view(self, request, callback, callback_args, callback_kwargs):
    print("CustomerMiddleware1 process_view")


    结果如下:

    
    
    
    
    


    注意:process_view如果有返回值,会越过其他的process_view以及视图函数,但是所有的process_response都还会执行。

      3、当自定义的中间件有process_view 且 当views出现错误时:

    from django.utils.deprecation import MiddlewareMixin
    from django.shortcuts import HttpResponse
    
    
    class CustomerMiddleware(MiddlewareMixin):
    
        def process_request(self,request):
            print("CustomerMiddleware1 process_request....")
    
            #return HttpResponse("forbidden....")
    
    
        def process_response(self,request,response):
            print("CustomerMiddleware1 process_response")
    
            return response
            #return HttpResponse("hello yuan")
    #
        def process_view(self, request, callback, callback_args, callback_kwargs):
            print("CustomerMiddleware1 process_view")
    # #
    #
        def process_exception(self, request, exception):
    
            print("CustomerMiddleware1 process_exception")
            return HttpResponse(exception)

     只需要掌握两个

    request

    response

     def process_request(self,request):
            print("CustomerMiddleware1 process_request....")
    
        
        def process_response(self,request,response):
            print("CustomerMiddleware1 process_response")
    
            return response #必须有返回值

    中间件应用之用户认证

    通过中间件实现装饰器 @login_required  用户认证

    完整代码实现

    url

    from django.contrib import admin
    from django.urls import path
    from app01 import views
    urlpatterns = [
        path('admin/', admin.site.urls),
        path('login/', views.login),
        path('index/', views.index),
        path('logout/', views.logout),
        path('reg/', views.reg),
        # path('set_pwd/', views.set_pwd),
        path('order/', views.order),
    ]

    views

    from django.shortcuts import render,redirect
    
    # Create your views here.
    from django.contrib import auth
    from django.contrib.auth.models import User
    from django.contrib.auth.decorators import login_required
    
    def login(request):
        if request.method=='POST':
            name=request.POST.get('user')
            pwd=request.POST.get('pwd')
            # 如果验证成功返回user对象,否则返回none
            user=auth.authenticate(username=name,password=pwd)
            print('user',user)
            if user:
                auth.login(request,user)  # request.user:当前登录对象
                # return redirect('/index/')
                # 加装饰器
    
                next_url = request.GET.get("next", "/index/")
                return redirect(next_url)
        return render(request,'login.html')
    
    #@login_required
    def index(request):
        # print('request.user',request.user)
        # if not request.user.is_authenticated:
        #     return redirect('/login/')
        # username=request.user
        return render(request,'index.html',locals())
    
    # 注销
    def logout(request):
        auth.logout(request)
        return redirect("/login/")
    
    # 注册
    def reg(request):
        if request.method == "POST":
            name = request.POST.get("user")
            pwd = request.POST.get("pwd")
            # User.objects.create(username=user,password=pwd)
            user = User.objects.create_user(username=name, password=pwd)
            return redirect("/login/")
        return render(request, "reg.html")
    
    #@login_required
    # 加装饰器后必须登录才能访问,为登录跳转到默认的页面
    def order(request):
    
        # if not request.user.is_authenticated:
        #     return redirect("/login/")
    
        return render(request,"order.html")

    settings.py

    
    
    STATIC_URL = '/static/'

    LOGIN_URL='/login/'

    #
    设置白名单 WHITE_LIST = ["/login/", "/reg/", "/logout/"]

    my_middlewares.py

    from django.utils.deprecation import MiddlewareMixin
    from django.shortcuts import redirect
    from authDemo import settings
    
    class AuthMiddleware(MiddlewareMixin):
    
        def process_request(self,request):
            white_list = settings.WHITE_LIST
            if request.path in white_list:
                return None
            if not request.user.is_authenticated:
                return redirect("/login/")

     设置成白名单的才可以访问,其他的不能访问;当用户登陆成功后,其他的视图就可以直接

    访问无需登录验证

    装饰器,中间件具体情况具体分析

    只有几个需要验证则装饰器

    需要大量验证的则中间件

    三、应用案例

    1、做IP访问频率限制

    某些IP访问服务器的频率过高,进行拦截,比如限制每分钟不能超过20次。

    2、URL访问过滤

    如果用户访问的是login视图(放过)
    如果访问其他视图,需要检测是不是有session认证,已经有了放行,没有返回login,这样就省得在多个视图函数上写装饰器了!
    作为延伸扩展内容,有余力的同学可以尝试着读一下以下两个自带的中间件:

    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.contrib.auth.middleware.AuthenticationMiddleware',
  • 相关阅读:
    BZOJ 3992: [SDOI2015]序列统计
    BZOJ 4836: [Lydsy1704月赛]二元运算
    2.Add Two Numbers
    [RN] React Native 调试技巧
    [商业世界] 商业三流:信息流、资金流、物流
    [杂谈] 记 程序员 对抗失眠烦恼的 大法
    [未来成长]让写作成为一种生活习惯
    [未来成长] 分享:《麦肯锡教我的写作武器》如何写出一篇具有逻辑表现力的文案
    [未来成长] 分享:在腾讯的八年,我的职业思考
    [PHP]:AES对称加密 -- 支持PHP7
  • 原文地址:https://www.cnblogs.com/foremostxl/p/9981896.html
Copyright © 2020-2023  润新知