Django 详解 中间件Middleware
Django中间件
还是涉及到django的请求生命周期。middle ware 请求穿过中间件到达url,再经过中间件返回给用户。
简单实例
django项目根目录新建一个Middle文件夹,再新建一个test.py文件

在test文件中写入;其中的类必须继承 from django.utils.deprecation import MiddlewareMixin
from django.utils.deprecation import MiddlewareMixin
class M1(MiddlewareMixin):
def process_request(self, request):
print('m1')
def process_response(self, request, response):
print('m1_r')
return response class M2(MiddlewareMixin):
def process_request(self, request):
print('m2')
def process_response(self, request, response):
print('m2_r')
return response class M3(MiddlewareMixin):
def process_request(self, request):
print('m3') def process_response(self, request, response):
print('m3_r')
return response
将你的测试中间件加入Django的中间件配置中,settings文件

随便建一组对应路由。

在index函数里面写上
def index(request):
print('到达')
return HttpResponse('ok')
查看结果:

此时如果给某个中间件的process_request返回一个HttpResponse:
#!/user/bin/env python
# -*-coding: utf-8-*-
from django.utils.deprecation import MiddlewareMixin
from django.shortcuts import HttpResponse
class M1(MiddlewareMixin):
def process_request(self, request):
print('m1')
def process_response(self, request, response):
print('m1_r')
return response class M2(MiddlewareMixin):
def process_request(self, request):
print('m2')
return HttpResponse('中断') # 这里返回
def process_response(self, request, response):
print('m2_r')
return response class M3(MiddlewareMixin):
def process_request(self, request):
print('m3') def process_response(self, request, response):
print('m3_r')
return response
查看下结果



可能你好像懂了,但是并没有,真正的中间件过程其实还有一个process_view.


自定制中间件的另一种方式
下面是最新的django2.1文档内的自定制中间件的另一种写法。
一个中间件工厂是可以被调用的,它接收一个可调用的get_response方法并返回一个中间件。中间件也是可调用的,它接收请求并返回响应,就像一个view视图
def simple_middleware(get_response):
# 一次性配置和初始化 def middleware(request):
# 在每个请求之前被执行的代码
#这个视图(和后面的中间件)被调用 response = get_response(request) # 在每个请求或者响应之后代码被执行
# the view is called. return response return middleware
或者他也可以被重写成一个实例可以被调用的类,像这个:
class SimpleMiddleware:
def __init__(self, get_response):
self.get_response = get_response
# 一次性配置和初始化 def __call__(self, request):
# 在每一次请求之前代码被执行
# 视图或者接下来的中间件被调用
response = self.get_response(request) # 在每个请求或者响应之后代码将会被执行
# the view is called. return response
可调用的get_response是由django提供的,可以是真实的视图函数或者也可以是链上的下一个中间件。当前的中间件不需要具体精确地知道或者关心它是什么,只是它代表接下来执行(到来的)是什么。
上面是一个小小的简化,可调用的get_response如果在中间件的最后一位,将不会是一个真实的视图函数,更可能是处理的一个外部包装方法,它关注于应用视图中间件,调用视图和合适的URL参数,并且使用template和exception中间件.
中间件可以生存在你的Python路径的任何地方。
__init__(get_response)
中间件工厂必须接收一个get_response参数,你也可以为中间件初始化一些全局的状态。记住下面的这几点
1、django只能使用get_response参数来初始化你的中间件,所以你不能定义__init__()来需要其他的参数
2、跟__call__()方法每次请求被调用不同,__init__()方法只会被调用一次,当web服务启动的时候
例如
class LoginMiddleware:
def __init__(self, get_response):
self.get_response = get_response @method_decorator(adminlogin)
def __call__(self, request):
if request.method == "PUT":
request.data = dict(urllib.parse.parse_qsl(request.body.decode()))
request.data.update(request.GET.dict())
elif request.method == "POST":
request.data = request.POST.copy().dict()
response = self.get_response(request)
return response
def performance(func):
def wrapper(request):
mark1 = time.time()
start = time.strftime("%Y-%m-%d %H:%M:%S", time.gmtime(mark1)) # 2018-09-28 03:42:08
response = func(request)
if response.status_code == 200 and request.path.split('/')[1] == 'api': # api url& response success
mark2 = time.time()
cost = mark2 - mark1
end = time.strftime("%Y-%m-%d %H:%M:%S", time.gmtime(mark2))
note = 'url: %s |%s | total cost: %s | start time: %s | end time: %s\r\n' % (request.path, request.GET, cost, start, end) with open(performance_file, 'a+') as obj:
obj.write(note)
return response
return wrapper class PerformanceMiddleware:
def __init__(self, get_response):
self.get_response = get_response @method_decorator(performance)
def __call__(self, request):
response = self.get_response(request)
return response
详情请查阅django(version2.1)文档:https://docs.djangoproject.com/en/2.1/topics/http/middleware/
Django 详解 中间件Middleware的更多相关文章
- Python 19 Django 详解
本节概要 Django详解 前言 有一部分原因是,确实djando的课程有点多:并且,最近又在研究利用python做数据分析时间上耽误了.所以楼主讲所有的课程全部重新观看了一遍,再来撰写博客,其实说起 ...
- 【laravel54】详解中间件
1.中间件定义:对http请求进行一层过滤,通过过滤才能继续执行请求 2.中间件方法handle方法参数详解: 其中参数的形式可以有多个,使用[,]进行分割. 3.路由中使用中间件: 3.1 中间件使 ...
- Django框架之中间件MiddleWare
Django中的中间件是一个轻量级.底层的插件系统,可以介入Django的请求和响应处理过程,修改Django的输入或输出.中间件的设计为开发者提供了一种无侵入式的开发方式,增强了Django框架的健 ...
- Django 学习之中间件Middleware
一.中间件介绍 中间件顾名思义,是介于request与response处理之间的一道处理过程,相对比较轻量级,并且在全局上改变django的输入与输出.因为改变的是全局,所以需要谨慎实用,用不好会影响 ...
- Django 详解 信号Signal
Django信号 Django中提供了“信号调度”,用于在框架执行操作时解耦.通俗来讲,就是一些动作发生的时候,信号允许特定的发送者去提醒一些接受者. Model signals pre_init # ...
- Django详解之models操作
D jango 模型是与数据库相关的,与数据库相关的代码一般写在 models.py 中,Django 支持 sqlite3, MySQL, PostgreSQL等数据库,只需要在settings.p ...
- Django详解之四、cookie和session
一.使用背景 思路 简单的后台管理:对人员的管理 1. 登录注册 2. 老师 班级管理 学院管理 3. 增删改查 开发: 1. 定义数据库表结构 a) 表结构关系 i. class classes(m ...
- Django 详解<二> 之url和view
Django URL(路由系统) RL配置(URLconf)就像Django 所支撑网站的目录.它的本质是URL模式以及要为该URL模式调用的视图函数之间的映射表:你就是以这种方式告诉Django,对 ...
- Django 详解
Django是一个开源的Web应用框架,由Python写成.采用MVC的软件设计模式,主要目标是使得开发复杂的.数据库驱动的网站变得简单.Django注重组件的重用性和“可插拔性”,敏捷开发和DRY法 ...
随机推荐
- ImageMagick: win7 | win8 & uac (用户帐户控制) 注册表的一些事
现在用win7,win8的人越来越多了, 程序在一些 win 7, win8 上运行会遇到一些之前没想过的兼容性问题. 比如 64位系统运行32位程序时的注册表重定向,还有因为 uac (用户帐户控制 ...
- okhttp 内网可以有,但外网访问数据返不回来,代码一样
:1.问题点在于 下图红框里写成 text/html了,需要改成application/json,造成的问题有:unexpected end of stream 这个是406错误:加上日志之后okh ...
- django_orm查询性能优化
查询操作和性能优化 1.基本操作 增 models.Tb1.objects.create(c1='xx', c2='oo') 增加一条数据,可以接受字典类型数据 **kwargs obj = mode ...
- Linux基础入门教程
Linux基础入门教程 --------- Linux学习路径 Linux学习者,常常不知道自己改怎么学习linux:Linux初级,也就是入门linux前提是需要有一些计算机硬件相关的知识或是有一下 ...
- bootstrap实现checkbox全选、取消全选
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <!-- 最新版本的 ...
- 关于PHP中会话技术的知识点分享
前言:在PHP中会话技术也是特别重要的,主要应用在免登录,保存一些持久化数据等等的方面,但是后期的介绍中,我将会放弃这种技术改用redis方法来替换这种方法. (一)cookie技术(即数据缓存在客户 ...
- jenkins笔记
java -jar jenkins.war -httpPort=9090 以9090端口启动jekins的web应用(内置jetty)
- 【1】[leetcode-124] 二叉树中的最大路径和
(没做出来,典型题目重要) 二叉树中的最大路径和(hard) 给定一个非空二叉树,返回其最大路径和. 本题中,路径被定义为一条从树中任意节点出发,达到任意节点的序列.该路径至少包含一个节点,且不一定经 ...
- c++进阶学习
以后可能要做c++开发了..记录要看的书和可能用的技术,让自己有个学习的方向... 1. 语言基础 2. 算法与数据结构基础 3. 多线程开发基础 4. 数据库 5. 网络编程 6. 内存数据库技 ...
- Could not find default endpoint element that references contract 'wcfXXXXXXXXXXX' in the ServiceMode
Service本身没有问题,但是调用的时候,只在DataAccessSilverlight里引用了,而在主工程WebGISDemo里没有引用服务PowerDataServiceReference,所以 ...