3分钟看懂Python后端必须知道的Django的信号机制


Posted in Python onJuly 26, 2020

概念

django自带一套信号机制来帮助我们在框架的不同位置之间传递信息。也就是说,当某一事件发生时,信号系统可以允许一个或多个发送者(senders)将通知或信号(signals)发送给一组接受者(receivers)。

(感觉就很像Qt的信号与槽机制)

信号系统包含以下三要素:

  • 发送者-信号的发出方
  • 信号-信号本身
  • 接收者-信号的接受者

Django内置了一整套信号,下面是一些比较常用的:

Django内置信号

Model signals
  pre_init          # django的modal执行其构造方法前,自动触发
  post_init          # django的modal执行其构造方法后,自动触发
  pre_save          # django的modal对象保存前,自动触发
  post_save          # django的modal对象保存后,自动触发
  pre_delete         # django的modal对象删除前,自动触发
  post_delete         # django的modal对象删除后,自动触发
  m2m_changed         # django的modal中使用m2m字段操作第三张表(add,remove,clear)前后,自动触发
  class_prepared       # 程序启动时,检测已注册的app中modal类,对于每一个类,自动触发
Management signals
  pre_migrate         # 执行migrate命令前,自动触发
  post_migrate        # 执行migrate命令后,自动触发
Request/response signals
  request_started       # 请求到来前,自动触发
  request_finished      # 请求结束后,自动触发
  got_request_exception    # 请求异常后,自动触发
Test signals
  setting_changed       # 使用test测试修改配置文件时,自动触发
  template_rendered      # 使用test测试渲染模板时,自动触发
Database Wrappers
  connection_created     # 创建数据库连接时,自动触发

在ORM模型的save()方法调用之前或之后发送信号

django.db.models.signals.pre_save
django.db.models.signals.post_save

在ORM模型或查询集的delete()方法调用之前或之后发送信号。

django.db.models.signals.pre_delete
django.db.models.signals.post_delete

当多对多字段被修改时发送信号。

django.db.models.signals.m2m_changed

当接收和关闭HTTP请求时发送信号。

django.core.signals.request_started
django.core.signals.request_finished

这就很方便了,想象一下这样一个情景,我们引入了一个第三方的Django库,他会创建模型,我们想要在不修改他的代码的情况下,在他创建模型的时候同时修改其他表的数据,那么这时候信号机制就能派上用场了。

简单使用例子
信号监听有两种方式,一种是 Signal.connect() 方法,一种是装饰器。

Signal.connect()方法

方法原型:

Signal.connect(receiver, sender=None, weak=True, dispatch_uid=None)[source]

参数:

  • receiver :当前信号连接的回调函数,也就是处理信号的函数。
  • sender :指定从哪个发送方接收信号。
  • weak : 是否弱引用
  • dispatch_uid :信号接收器的唯一标识符,以防信号多次发送。

装饰器法

from django.core.signals import request_finished
from django.dispatch import receiver

@receiver(request_finished)
def my_callback(sender, **kwargs):
  print("Request finished!")

可以给 receiver 添加 sender 参数来筛选接收哪个模型产生的信号,比如:

from django.db.models import signals
from django.dispatch import receiver

@receiver(signals.post_save, sender=MyModel)
def demo_callback(sender: MyModel, instance: MyModel, **kwargs):
  pass

注意要读取传入的模型对象一定是要用 instance 参数,不是 sender ! sender 是一个类型!

设置dispatch_uid参数防止重复信号
为了防止重复信号,可以设置dispatch_uid参数来标识你的接收器,标识符通常是一个字符串,如下所示:

from django.core.signals import request_finished

request_finished.connect(my_callback, dispatch_uid="my_unique_identifier")

断开信号

在满足某些条件下,我们不再需要监听信号,可以选择断开信号。

Signal.disconnect()用来断开信号的接收器。和Signal.connect()中的参数相同。如果接收器成功断开,返回True,否则返回False。

Signal.disconnect(receiver=None, sender=None, dispatch_uid=None)[source]

后记

最近刚入职项目很多特别的忙,也学到了很多新技术,但是感觉已经好久没有时间来好好写一篇博客了,就算是本文也是短时间拼凑的,真是难呀,周末得好好整理一下最近的技术笔记。

关于Django信号机制的更多内容(自定义信号等),可以参考官方文档。 https://docs.djangoproject.com/en/3.0/topics/signals/

到此这篇关于3分钟看懂Python后端必须知道的Django的信号机制的文章就介绍到这了,更多相关Django 信号机制内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
python利用hook技术破解https的实例代码
Mar 25 Python
Python+Socket实现基于TCP协议的客户与服务端中文自动回复聊天功能示例
Aug 31 Python
python交互式图形编程实例(一)
Nov 17 Python
python基础教程项目四之新闻聚合
Apr 02 Python
对numpy的array和python中自带的list之间相互转化详解
Apr 13 Python
pytorch permute维度转换方法
Dec 14 Python
Python数据类型之String字符串实例详解
May 08 Python
python提取log文件内容并画出图表
Jul 08 Python
详解Django定时任务模块设计与实践
Jul 24 Python
python如何操作mysql
Aug 17 Python
浅谈Python从全局与局部变量到装饰器的相关知识
Jun 21 Python
Django框架中表单的用法
Jun 10 Python
给Django Admin添加验证码和多次登录尝试限制的实现
Jul 26 #Python
python如何支持并发方法详解
Jul 25 #Python
Python获取浏览器窗口句柄过程解析
Jul 25 #Python
用Python 爬取猫眼电影数据分析《无名之辈》
Jul 24 #Python
基于Python实现简单学生管理系统
Jul 24 #Python
用python实现学生管理系统
Jul 24 #Python
Python 解析简单的XML数据
Jul 24 #Python
You might like
怎样在php中使用PDF文档功能
2006/10/09 PHP
推荐php模板技术[转]
2007/01/04 PHP
图解找出PHP配置文件php.ini的路径的方法
2014/08/20 PHP
php递归json类实例
2014/12/02 PHP
PHP实现微信退款的方法示例
2019/03/26 PHP
PHP实现基于状态的责任链审批模式详解
2019/05/31 PHP
JavaScript delete 属性的使用
2009/10/08 Javascript
js实现广告漂浮效果的小例子
2013/07/02 Javascript
javascript跨域原因以及解决方案分享
2015/04/08 Javascript
Javascript获取图片原始宽度和高度的方法详解
2016/09/20 Javascript
关于微信上网页图片点击全屏放大效果
2016/12/19 Javascript
vue的事件绑定与方法详解
2017/08/16 Javascript
JavaScript复制内容到剪贴板的两种常用方法
2018/02/27 Javascript
js针对图片加载失败的处理方法分析
2019/08/24 Javascript
微信小程序 this.triggerEvent()的具体使用
2019/12/10 Javascript
[02:32]【DOTA2亚洲邀请赛】iceice,梦开始的地方
2017/03/13 DOTA
利用Python实现颜色色值转换的小工具
2016/10/27 Python
基于Django与ajax之间的json传输方法
2018/05/29 Python
程序员写Python时的5个坏习惯,你有几条?
2018/11/26 Python
Python facenet进行人脸识别测试过程解析
2019/08/16 Python
Tensorflow tf.nn.depthwise_conv2d如何实现深度卷积的
2020/04/20 Python
python能自学吗
2020/06/18 Python
python中count函数知识点浅析
2020/12/17 Python
Python3.9.0 a1安装pygame出错解决全过程(小结)
2021/02/02 Python
通过css3动画和opacity透明度实现呼吸灯效果
2019/08/09 HTML / CSS
番木瓜健康和保健产品第一大制造商:Herbal Papaya
2017/04/25 全球购物
英国鹦鹉店:Parrot Essentials
2018/12/03 全球购物
教师师德教育的自我评价
2013/10/31 职场文书
机电专业大学生职业规划书范文
2014/02/25 职场文书
经典促销广告词大全
2014/03/19 职场文书
央视元宵晚会主持串词
2014/03/25 职场文书
教师党员承诺书2015
2015/01/21 职场文书
交心谈心活动总结
2015/05/11 职场文书
Java后台生成图片的完整步骤
2021/08/04 Java/Android
OpenCV项目实践之停车场车位实时检测
2022/04/11 Python
《金肉人》米特&《航海王》阿鹤声优松岛实因胰脏癌去世 享寿81岁
2022/04/13 日漫