在学习django中间件之前,先来认识一下django的生命周期,如下图所示:

django生命周期:浏览器发送的请求会先经过wsgiref模块处理解析出request(请求数据)给到中间件,然后通过路由控制执行对应的视图函数,从而和模板,db进行交互,交互完的数据再通过视图函数返回给中间件,最后wsgiref模块将返回的数据封装成http形式的数据给到浏览器并进行展示。

了解了django的生命周期后,我们就可以开始着手写一个自己的中间件了,接下来认识几个常用的中间件方法

1.process_request

单个中间件

首先在app下创建一个py文件,定义你的中间件类名

from django.utils.deprecation import MiddlewareMixin
from django.http import HttpResponse class MiddlewareShow(MiddlewareMixin): def process_request(self, request): print('MiddlewareShow1')

然后将你的py文件路径写入django主项目的settings的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',
'proxy_pro.middleware.MiddlewareShow' # 此条是新加的 从该项目路径开始写
]

然后执行一个视图函数即可,查看控制台打印

此时自己创建的process_request方法就生效了

多个中间件

此时在之前的py文件中再新建一个类

class MiddlewareShow(MiddlewareMixin):

    def process_request(self, request):
print('MiddlewareShow1 Request') class MiddlewareShowTwo(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow2 Request')

然后将新建类的路径也放在django主项目的settings的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',
'proxy_pro.middleware.MiddlewareShow',
'proxy_pro.middleware.MiddlewareShowTwo'
]

执行视图函数,查看控制台打印

此时定义的两个中间件都生效了,执行顺序是先1后2

 2.process_response

分别在刚才的类中添加response方法

from django.utils.deprecation import MiddlewareMixin
from django.http import HttpResponse class MiddlewareShow(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow1 Request') def process_response(self, request, response):
print('MiddlewareShow1 Response')
return response class MiddlewareShowTwo(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow2 Request') def process_response(self, request, response):
print('MiddlewareShow2 Response')
return response

执行视图函数,查看控制台打印

此时可以看出请求先依次执行了中间件的Request,然后再去执行视图函数,返回是先执行跟后面的中间件再依次往前

这时突然冒出一个想法,如果在request时直接返回了某个东西,还会继续去执行后面的视图函数吗?那我们就来测试一下

from django.utils.deprecation import MiddlewareMixin
from django.http import HttpResponse class MiddlewareShow(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow1 Request')
return HttpResponse('request1时已返回') def process_response(self, request, response):
print('MiddlewareShow1 Response')
return response class MiddlewareShowTwo(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow2 Request') def process_response(self, request, response):
print('MiddlewareShow2 Response')
return response

执行视图函数,查看控制台

此时可以看出当request中有返回时,直接不执行后面的内容了

3.process_view

分别在刚才的py文件中添加view方法

from django.utils.deprecation import MiddlewareMixin
from django.http import HttpResponse class MiddlewareShow(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow1 Request') def process_response(self, request, response):
print('MiddlewareShow1 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow1 process_view') class MiddlewareShowTwo(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow2 Request') def process_response(self, request, response):
print('MiddlewareShow2 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow2 process_view')

执行视图函数,查看控制台返回

此时可以看出这个process_view方法会在执行完request后执行,执行完再去执行视图函数

这时候就有疑问了,那这个有什么用呢?我们先把它里面的参数打印一下

from django.utils.deprecation import MiddlewareMixin
from django.http import HttpResponse class MiddlewareShow(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow1 Request') def process_response(self, request, response):
print('MiddlewareShow1 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow1 process_view')
print('=====>callback', callback)
print('=====>callback_args', callback_args) class MiddlewareShowTwo(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow2 Request') def process_response(self, request, response):
print('MiddlewareShow2 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow2 process_view')

执行视图函数打印结果

此时可以看到这个view里面的参数callback是视图方法,callback_args是请求参数,那我们试着去请求下看看

class MiddlewareShow(MiddlewareMixin):

    def process_request(self, request):
print('MiddlewareShow1 Request') def process_response(self, request, response):
print('MiddlewareShow1 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow1 process_view')
print('=====>callback', callback)
print('=====>callback_args', callback_args)
ret = callback(callback_args) # 请求
return ret class MiddlewareShowTwo(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow2 Request') def process_response(self, request, response):
print('MiddlewareShow2 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow2 process_view')

查看打印结果

结果显然易见,请求视图函数成功了,返回了以后就没有去执行process_view2了,这边的作用就是可以拦截请求

4.process_exception

分别在刚才的py文件中添加exception方法

from django.utils.deprecation import MiddlewareMixin
from django.http import HttpResponse class MiddlewareShow(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow1 Request') def process_response(self, request, response):
print('MiddlewareShow1 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow1 process_view')
# print('=====>callback', callback)
# print('=====>callback_args', callback_args)
# ret = callback(callback_args)
# return ret def process_exception(self, request, exception):
print('MiddlewareShow1 process_exception')
return HttpResponse(exception) class MiddlewareShowTwo(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow2 Request') def process_response(self, request, response):
print('MiddlewareShow2 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow2 process_view') def process_exception(self, request, exception):
print('MiddlewareShow2 process_exception')
return HttpResponse(exception)

执行视图函数,查看控制台

问题来了,为啥没有走这个中间件方法呢?别慌,我们在视图函数中加个错

def middle_show(request):
leo print('执行了视图函数') return HttpResponse('hhh')

此时在执行下看看

此时可以看到当process_exception2获取到报错后,就返回了没有执行process_exception1

那如果我们在process_exception1处理呢,我们来测试下

from django.utils.deprecation import MiddlewareMixin
from django.http import HttpResponse class MiddlewareShow(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow1 Request') def process_response(self, request, response):
print('MiddlewareShow1 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow1 process_view')
# print('=====>callback', callback)
# print('=====>callback_args', callback_args)
# ret = callback(callback_args)
# return ret def process_exception(self, request, exception):
print('MiddlewareShow1 process_exception')
return HttpResponse(exception) class MiddlewareShowTwo(MiddlewareMixin): def process_request(self, request):
print('MiddlewareShow2 Request') def process_response(self, request, response):
print('MiddlewareShow2 Response')
return response def process_view(self, request, callback, callback_args, callback_kwargs):
print('MiddlewareShow2 process_view') def process_exception(self, request, exception):
print('MiddlewareShow2 process_exception')

执行视图函数,打印看下

此时还会走到process_exception1哦,然后把错误返回给页面。

以上就是django中间件的介绍,希望和大家多多学习!转载请说明出处,尊重劳动成果!!!

django中间件介绍的更多相关文章

  1. python Django 中间件介绍

    我们一直都在使用中间件,只是没有注意到而已,打开Django项目的Settings.py文件,看到下面的MIDDLEWARE配置项,django默认自带的一些中间件: MIDDLEWARE = [ ' ...

  2. {Django基础九之中间件} 一 前戏 二 中间件介绍 三 自定义中间件 四 中间件的执行流程 五 中间件版登陆认证

    Django基础九之中间件 本节目录 一 前戏 二 中间件介绍 三 自定义中间件 四 中间件的执行流程 五 中间件版登陆认证 六 xxx 七 xxx 八 xxx 一 前戏 我们在前面的课程中已经学会了 ...

  3. 【12】Django 中间件

     前戏 我们在前面的课程中已经学会了给视图函数加装饰器来判断是用户是否登录,把没有登录的用户请求跳转到登录页面.我们通过给几个特定视图函数加装饰器实现了这个需求.但是以后添加的视图函数可能也需要加上装 ...

  4. Django中间件2

    前戏 我们在前面的课程中已经学会了给视图函数加装饰器来判断是用户是否登录,把没有登录的用户请求跳转到登录页面.我们通过给几个特定视图函数加装饰器实现了这个需求.但是以后添加的视图函数可能也需要加上装饰 ...

  5. 自定义Django中间件(登录验证中间件实例)

    前戏 我们在前面的课程中已经学会了给视图函数加装饰器来判断是用户是否登录,把没有登录的用户请求跳转到登录页面.我们通过给几个特定视图函数加装饰器实现了这个需求.但是以后添加的视图函数可能也需要加上装饰 ...

  6. Django 2.0 学习(20):Django 中间件详解

    Django 中间件详解 Django中间件 在Django中,中间件(middleware)其实就是一个类,在请求到来和结束后,Django会根据自己的规则在合适的时机执行中间件中相应的方法. 1. ...

  7. Django中间件的5种自定义方法

    阅读目录(Content) Django中间件 自定义中间件 中间件(类)中5种方法 中间件应用场景 回到顶部(go to top) Django中间件 在http请求 到达视图函数之前   和视图函 ...

  8. Django 中间件使用

     前戏 我们在前面的课程中已经学会了给视图函数加装饰器来判断是用户是否登录,把没有登录的用户请求跳转到登录页面.我们通过给几个特定视图函数加装饰器实现了这个需求.但是以后添加的视图函数可能也需要加上装 ...

  9. Django中间件如何处理请求

    Django中间件 在http请求 到达视图函数之前   和视图函数return之后,django会根据自己的规则在合适的时机执行中间件中相应的方法. Django1.9版本以后中间件的执行流程 1. ...

随机推荐

  1. scala 两个map合并,key相同时value相加/相减都可

    scala 两个map合并,key相同时value相加 1.map自带的合并操作 2.map函数 2.1示例 2.2合并两个map 3.用foldLeft 3.1 语法 3.2 合并两个map 1.m ...

  2. Spark练习之action操作开发

    Spark练习之action操作开发 一.reduce 1.1 Java 1.2 Scala 二.collect 2.1 Java 2.2 Scala 三.count 3.1 Java 3.2 Sca ...

  3. Cisco的互联网络操作系统IOS和安全设备管理器SDM__路由器软、硬件知识

    路由器软.硬件知识 1.路由器的组件: 组件 解释 Bootstrap 存储在ROM中的微代码,bootstrap用于在初始化阶段启动路由器.它将启动路由器然后装入IOS POST(开机自检) 存储在 ...

  4. C#委托的进一步学习

    一.委托的说明 namespace LearningCsharp { class Program { //定义一个委托,使用delegate加上方法签名 //将委托理解为存储方法的"数组&q ...

  5. k8s 调度 GPU

    最近公司有项目想在 k8s 集群中运行 GPU 任务,于是研究了一下.下面是部署的步骤. 1. 首先得有一个可以运行的 k8s 集群. 集群部署参考 kubeadm安装k8s 2. 准备 GPU 节点 ...

  6. 要习惯用vector代替数组

    cin>>n>>m; vector<int>a(n),b(m); 或者: vector<int>a(n,0),b(m,0);

  7. 【noi 2.7_413】Calling Extraterrestrial Intelligence Again(算法效率--线性筛素数+二分+测时)

    题意:给3个数M,A,B,求两个质数P,Q.使其满足P*Q<=M且A/B<=P/Q<=1,并使P*Q最大.输入若干行以0,0,0结尾. 解法:先线性筛出素数表,再枚举出P,二分出对应 ...

  8. 2020ICPC&#183;小米 网络选拔赛第一场 J.Matrix Subtraction (贪心,二维差分)

    题意:给一个\(nXm\)的矩阵,可以选取\(aXb\)的子矩阵,使子矩阵中的所有元素减一,问最后是否能使矩阵中所有元素变为\(0\). 题解:首先贪心,我们看最左上角的元素,如果\(g[1][1]\ ...

  9. Codeforces Round #613 (Div. 2) B. Just Eat It! (DP)

    题意:有一个长度为\(n\)的序列,找出最大的长度不为\(n\)的子段和,问最大子段和是否小于所有元素和. 题解:最大子段和我们可以直接用dp来找,每次状态转移为:\(dp[i]=max(dp[i-1 ...

  10. NFS 共享存储

    目录 环境准备 NFS服务端 NFS客户端 部署时常见报错 httpd服务 NFS 共享存储的坑 环境准备 主机名 WanIP(Wide Area Network) LanIP(Local Area ...