Django元件---Django請求生命週期和中介軟體

Bound_w發表於2018-12-21

Django元件---Django請求生命週期和中介軟體

Django請求生命週期

      

說明:

  • client代表瀏覽器,瀏覽器的內部為我們封裝了socket,Django的WSGI模組也為我們封裝了socket
  • 當使用者在瀏覽器輸入URL或者點選某一個連結的時候,瀏覽器會生成請求頭(get)和請求體(post)傳送給伺服器
  • 請求到達Django的伺服器之後,首先WSGI會根據http請求將請求的資訊解包,然後封裝到HttpRequest中,
  • 再依次通過Django的中介軟體proces_request方法(Django自帶7箇中介軟體,每個中介軟體都有一個類,,每個類中都有process_request的方法)
  • 然後通過URL控制器分發後,執行對應的檢視,檢視中可以根據對應的請求拿到自己需要的資料,拿到的資料通過render渲染的方式,傳送給伺服器
  • 返回的時候也會依次經過中介軟體的process_request方法
  • 在經過WSGI模組WSGI根據http響應協議的格式對響應訊息進行封裝,然後返回給客戶端(瀏覽器)
  • 客戶端瀏覽器收到響應的資料,經過渲染,顯示給使用者

中介軟體的概念

  中介軟體是介於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物件,這些都可以通過中介軟體(middleware)來實現。可能你還想在view執行之前做一些操作,這種情況就可以用 middleware來實現。

django預設的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',
]

自定義中介軟體(中介軟體一共有四種方法)

process_request

process_view

process_response

process_exception

#process_request和process_response

解釋:

  當使用者發起請求的時候會依次經過所有的中介軟體,請求的process_request執行到view的函式中,view函式處理過後,再一次穿過中介軟體,process_response會最後返回給使用者.

這些都是django 的中介軟體,我們也可以自己定義中介軟體

自定義中介軟體:

1.書寫一個類,必須繼承MiddlewareMinx,但是需要匯入from django.utils.deprecation import MiddlewareMixin

 

具體步驟:

1.檢視(view)

def index(request):

  print("view函式")

  return  HttpResponse("ok")

2.中介軟體Mymiddlewares.py

from django.utils.deprecation import MiddlewareMixin
from django.shortcuts import HttpResponse

class Md1(MiddlewareMixin):

    def process_request(self,request):
        print("Md1請求")
 
    def process_response(self,request,response):
        print("Md1返回")
        return response

class Md2(MiddlewareMixin):

    def process_request(self,request):
        print("Md2請求")
        #return HttpResponse("Md2中斷")
    def process_response(self,request,response):
        print("Md2返回")
        return response

結果:

Md1請求
Md2請求
view函式...
Md2返回
Md1返回

注意:如果當請求到達請求2的時候直接不符合條件返回,即return HttpResponse("Md2中斷"),程式將把請求直接發給中介軟體2返回,然後依次返回到請求者,結果如下:

返回Md2中斷的頁面,後臺列印如下:

Md1請求
Md2請求
Md2返回
Md1返回

流程圖如下:

 

 process_view

  process_view(self, request, callback, callback_args, callback_kwargs)

  Mymiddlewares.py修改如下:

from django.utils.deprecation import MiddlewareMixin
     from django.shortcuts import HttpResponse

        class Md1(MiddlewareMixin):
            def process_request(self,request):
                print("Md1請求")
                # return HttpResponse("Md1中斷")
            def process_response(self,request,response):
                print("Md1返回")
                return response
            def process_view(self, request, callback, callback_args, callback_kwargs):
                print("Md1view")

        class Md2(MiddlewareMixin):
            def process_request(self,request):
                print("Md2請求")
                # return HttpResponse("Md2中斷")
            def process_response(self,request,response):
                print("Md2返回")
                return response
            def process_view(self, request, callback, callback_args, callback_kwargs):
                print("Md2view")

 列印結果如下:

Md1請求
  Md2請求
  Md1view
  Md2view
  view函式...
  Md2返回
  Md1返回

分析上面的過程,如下面流程圖:

      

當最後一箇中間的process_request到達路由關係對映之後,返回到中

process_exception(self, request, exception)

 

間件1的process_view,然後依次往下,到達views函式,最後通過process_response依次返回到達使用者。

class Md1(MiddlewareMixin):
            def process_request(self,request):
                print("Md1請求")
            def process_response(self,request,response):
                print("Md1返回")
                return response
            def process_view(self, request, callback, callback_args, callback_kwargs):
                print("callback ") # 訪問index時列印<function index at 0x0000000003EF48C8>
                # 說明callback就是當前請求url對應的檢視函式名
                response = callback(request,*callback_args,**callback_kwargs)
                return response

注意:process_view如果有返回值,會越過其他的process_view以及檢視函式,但是所有的process_response都還會執行。

3、process_exception

流程圖如下:

      

Mymiddlewares.py修改如下:

class Md1(MiddlewareMixin):
            def process_request(self,request):
                print("Md1請求")
            def process_response(self,request,response):
                print("Md1返回")
                return response
            def process_view(self, request, callback, callback_args, callback_kwargs):
                # response=callback(request,*callback_args,**callback_kwargs)
                # return response
                print("md1 process_view...")
            def process_exception(self,request, exception):
                print("md1 process_exception...")

        class Md2(MiddlewareMixin):
            def process_request(self,request):
                print("Md2請求")
            def process_response(self,request,response):
                print("Md2返回")
                return response
            def process_view(self, request, callback, callback_args, callback_kwargs):
                print("md2 process_view...")
            def process_exception(self, request, exception):
                print("md2 process_exception...")
                return HttpResponse(exception)

當檢視函式出現錯誤時,列印結果如下(頁面接收到錯誤型別):

Md1請求
    Md2請求
    md1 process_view...
    md2 process_view...
    view函式...
    md2 process_exception...
    Md2返回
    Md1返回

應用案例:

1、做IP訪問頻率限制

       某些IP訪問伺服器的頻率過高,進行攔截,比如限制每分鐘不能超過20次。

2、URL訪問過濾

相關文章