今天女朋友去面試,作為一個(gè)python小白的她被問到中間件后一臉懵逼,回來后抓緊讓我給她惡補(bǔ)了一下。
中間件,故名思議,是在中間的一個(gè)插件。那么是在什么中間呢。
是在:

中間件的作用就是:在httpRequest請(qǐng)求還沒到的view之前,與view返回的httpResponse還未發(fā)送給請(qǐng)求者之前,對(duì)httpRequest與httpResponse做出修改。
(因?yàn)楸救瞬恢肋@篇文章不知道怎么布局,所以全程參考官方文檔,有興趣的同學(xué)可以直接看官方文檔:https://docs.djangoproject.com/en/1.8/topics/http/middleware/)
激活中間件
Django的setting文件中有一個(gè)列表專門放置中間件:
MIDDLEWARE_CLASSES = [
'django.middleware.security.SecurityMiddleware',
'django.contrib.sessions.middleware.SessionMiddleware',
'django.middleware.common.CommonMiddleware',
'django.middleware.csrf.CsrfViewMiddleware',
'django.contrib.auth.middleware.AuthenticationMiddleware',
'django.contrib.auth.middleware.SessionAuthenticationMiddleware',
'django.contrib.messages.middleware.MessageMiddleware',
'django.middleware.clickjacking.XFrameOptionsMiddleware',
'middleware.customMiddleware',#這是我自定義的
'middleware.customMiddleware2'#這是我自定義的
]
只要把中間件放入MIDDLEWARE_CLASSES中就可以使用。這個(gè)MIDDLEWARE_CLASSES可以為空,但是Django官方文檔強(qiáng)烈建議至少保留CommonMiddleware,同時(shí)MIDDLEWARE_CLASSES中的中間件執(zhí)行是有順序的,例如:AuthenticationMiddleware會(huì)把授權(quán)用戶(authenticated user)保存到session中,所以它必須放在SessionMiddleware的后面。
鉤子與執(zhí)行順序
在http請(qǐng)求階段,在view調(diào)用之前,django會(huì)把MIDDLEWARE_CLASSES中定義的中間件從上到下挨個(gè)執(zhí)行一遍。下面是兩個(gè)鉤子函數(shù):
process_request()-
process_view()
在http返回階段,在調(diào)用過view之后,MIDDLEWARE_CLASSES列表中的中間件將會(huì)被從底向上執(zhí)行一遍。下面是三個(gè)鉤子函數(shù): -
process_exception()(only if the view raised an exception) -
process_template_response()(only for template responses) -
process_response()
image.png
就像一個(gè)洋蔥,每一個(gè)中間件就是洋蔥的一層,而view被一層一層的包裹在中間件的里面。
編寫自己的中間件
每一個(gè)中間件都是一個(gè)python類,繼承object,實(shí)現(xiàn)以下一個(gè)或多個(gè)方法。
process_request()
process_request(request)
參數(shù)為HttpRequest對(duì)象,在到達(dá)view之前,process_request(request)函數(shù)依次被執(zhí)行。process_request(request)函數(shù)的返回值為None或者HttpResponse,如果反回None,則繼續(xù)執(zhí)行剩下中間件的process_request(request),如果所有process_request(request)都返回None,則順序執(zhí)行中間件的 process_view() ,最后到的view。
返回None例子:
class customMiddleware(object):
def process_request(self,request):
print "process_request"
def process_view(self,request,call_back,callback_args, callback_kwargs):
print "process_view"
class customMiddleware2(object):
def process_request(self,request):
print "process_request2"
def process_view(self,request,call_back,callback_args, callback_kwargs):
print "process_view2"
輸出結(jié)果:


返回httpResponse例子:
from django.http import HttpResponse
class customMiddleware(object):
def process_request(self,request):
print "process_request"
response = HttpResponse("Here's the text of the Web page.")
return response
def process_view(self,request,call_back,callback_args, callback_kwargs):
print "process_view"
class customMiddleware2(object):
def process_request(self,request):
print "process_request2"
def process_view(self,request,call_back,callback_args, callback_kwargs):
print "process_view2"
輸出結(jié)果:


process_view
process_view(request, view_func, view_args, view_kwargs)
request是httpRequest對(duì)象,view_func是函數(shù)對(duì)象,并不是函數(shù)名稱字符串,view_args,view_kwargs是傳給view_func的參數(shù)。
例如:
urls.py
from temp.views import tempView
urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^login/(?P<activate_code>.*)/$', tempView.as_view(), name='temp'),
]
temp/view.py
from django.views.generic.base import View
# Create your views here.
class tempView(View):
def get(self,request,activate_code):
print('tempView')
middleware.py
class customMiddleware(object):
def process_request(self,request):
print "process_request"
def process_view(self,request,call_back,callback_args, callback_kwargs):
print call_back
print(callback_args)
print(callback_kwargs)
print "process_view"
輸入:

輸出:

process_view()執(zhí)行在view之前。
process_view()返回值也為None或者HttpResponse,如果為None就繼續(xù)向下一個(gè)中間件執(zhí)行process_view(),如果返回HttpResponse則直接返回,參見process_request()
process_template_response
process_template_response(request,response)
request為httpRequest對(duì)象,response為view返回,或者中間件返回的 TemplateResponse對(duì)象(TemplateResponse對(duì)象可以參見:https://blog.csdn.net/wizardforcel/article/details/48105085)。
process_template_response()函數(shù)在包含render() 方法的view函數(shù)執(zhí)行完以后執(zhí)行。同時(shí)process_template_response()函數(shù)可以返回一個(gè)TemplateResponse去重定向返回的頁面。
小栗子:
temp/view.py
from django.views.generic.base import View
from django.template.response import TemplateResponse
# Create your views here.
class tempView(View):
def get(self,request,activate_code):
print('tempView')
t = TemplateResponse(request, 'login.html', {},{},'503')
return t.render()
middleware.py
from django.template.response import TemplateResponse
class customMiddleware(object):
def process_template_response(self, request, response):
print 'process_template_response'
t = TemplateResponse(request, 'login2.html', {}, {}, '503')
return t.render()
login.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>測(cè)試</title>
</head>
<body>
</body>
</html>
login2.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
哈哈哈 我是login2,不是login
</body>
</html>
運(yùn)行結(jié)果:


process_response
process_response(request,response)
process_response()函數(shù)的參數(shù)分別問HttpRequest對(duì)象與HttpResponse對(duì)象或者StreamingHttpResponse對(duì)象。
process_response()總是會(huì)被執(zhí)行,并且返回一個(gè)HttpResponse或者StreamingHttpResponse,效果參見process_template_response()函數(shù)。
小栗子
test/view.py
from django.views.generic.base import View
class tempView(View):
def get(self,request,activate_code):
print('tempView')
middleware.py
from django.shortcuts import render
class customMiddleware(object):
def process_response(self, request, response):
print 'process_response'
return render(request, "login.html")
login.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>測(cè)試</title>
</head>
<body>
</body>
</html>
輸出:


process_exception
process_exception(request,response)
只有當(dāng)view拋出異常的時(shí)候才會(huì)觸發(fā)這個(gè)函數(shù)
不多說 上例子
temp/view.py
from django.views.generic.base import View
class tempView(View):
def get(self,request,activate_code):
raise Warning
print('tempView')
middleware.py
class customMiddleware(object):
def process_exception(self, request, exception):
print 'process_exception'
class customMiddleware2(object):
def process_exception(self, request, exception):
print 'process_exception2'
輸出:

