深入理解Django自定义信号(signals)


Posted in Python onOctober 15, 2018

django中自定义了一些singals,用于监听一些操作,并发出通知

官方解释:

Django 提供一个“信号分发器”,允许解耦的应用在框架的其它地方发生操作时会被通知到。

简单来说,信号允许特定的sender通知一组receiver某些操作已经发生。这在多处代码和同一事件有关联的情况下很有用。
django中已经内置了一些singals,在django/db/models/signal.py中,如

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     # 创建数据库连接时,自动触发

用法:

利用这几个singals可以实现model中的一些联动操作,比如,要想更改通过model更新记录时,记下操作者的日志,可以直接在操作的地方使用post_save装饰器,

或者改写post_save,使其记录相关信息,一劳永逸。或者在request请求时,记录请求信息。

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

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

如何自定义singals?

a. 定义singal文件

import django.dispatch
pizza_done = django.dispatch.Signal(providing_args=["toppings", "size"])

b. 注册singal

def callback(sender, **kwargs):
  print("callback")
  print(sender,kwargs)
  pizza_done.connect(callback)

c. 触发信号

from 路径 import pizza_done
pizza_done.send(sender='seven',toppings=123, size=456)

需求场景:

项目中有一个需求,当model(即库的数据)被修改或者删除时,自动触发一个redis的同步任务(后来发现这个需求没有意义....),model的保存有post_save,删除有post_delete,唯独没有update,而代码中使用update的场景蛮多的,就搜了下为什么就是没有update的singals。

看到:https://code.djangoproject.com/ticket/12184

其实很早就有人给django官方提过这种方式,为什么不在官方版本中添加,具体这个pr为什么没有被接受,可以看下里面的讨论,反正当时的django1.9仍然不支持,只能自己先写一个用用,有问题了再撤掉好了。

解决方式:

singals.py文件

# coding:utf-8
from django.dispatch import Signal
post_update = Signal(providing_args=["user"])

models.py文件

-----------针对某个model,重写其queryset中的update方法-----------

//引入自定义的signal文件
from tools import signals 

class MyCustomQuerySet(models.query.QuerySet):
  def update(self, **kwargs):
    super(MyCustomQuerySet, self).update(**kwargs)
    //update被调用时, 发送该singalsignals
    signals.post_update.send(sender=self.model, user="xxx")
    print("finished!")

class MyCustomManager(models.Manager):
  def get_queryset(self):
    return MyCustomQuerySet(self.model, using=self._db)

class crontab_ping(models.Model):
  name = models.CharField(max_length=64, blank=True, null=True)
  objects = MyCustomManager()

callback.py文件:

-------接收signal,触发操作----------

from tools.signals import post_update

@receiver(post_update)
def post_update_callback(sender, **kwargs):
  print(kwargs['user'])
  print("post_update_success")

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python多重继承新算法C3介绍
Sep 28 Python
自己编程中遇到的Python错误和解决方法汇总整理
Jun 03 Python
Python实现视频下载功能
Mar 14 Python
python下10个简单实例代码
Nov 15 Python
python2.7+selenium2实现淘宝滑块自动认证功能
Feb 24 Python
Python3实现将本地JSON大数据文件写入MySQL数据库的方法
Jun 13 Python
python+selenium 定位到元素,无法点击的解决方法
Jan 30 Python
python笔记之mean()函数实现求取均值的功能代码
Jul 05 Python
python线程定时器Timer实现原理解析
Nov 30 Python
Python3自动生成MySQL数据字典的markdown文本的实现
May 07 Python
基于python计算滚动方差(标准差)talib和pd.rolling函数差异详解
Jun 08 Python
Python+Opencv实现把图片、视频互转的示例
Dec 17 Python
使用numba对Python运算加速的方法
Oct 15 #Python
浅谈解除装饰器作用(python3新增)
Oct 15 #Python
python  创建一个保留重复值的列表的补码
Oct 15 #Python
python 美化输出信息的实例
Oct 15 #Python
python  Django中的apps.py的目的是什么
Oct 15 #Python
使用Python监视指定目录下文件变更的方法
Oct 15 #Python
Python调用adb命令实现对多台设备同时进行reboot的方法
Oct 15 #Python
You might like
东方红 - 来复式再生机的修复
2021/03/02 无线电
PHP原理之异常机制深入分析
2010/08/08 PHP
解决文件名解压后乱码的问题 将文件名进行转码的代码
2012/01/10 PHP
PHP文件上传主要代码讲解
2013/09/30 PHP
在Mac OS上自行编译安装Apache服务器和PHP解释器
2015/12/24 PHP
Yii框架防止sql注入,xss攻击与csrf攻击的方法
2016/10/18 PHP
php脚本守护进程原理与实现方法详解
2017/07/20 PHP
Open and Print a Word Document
2007/06/15 Javascript
JavaScript的parseInt 进制问题
2009/05/07 Javascript
Domino中运用jQuery读取视图内容的方法
2009/10/21 Javascript
选择复选框按钮置灰否则按钮可用
2014/05/22 Javascript
JavaScript中的console.trace()函数介绍
2014/12/29 Javascript
Bootstrap每天必学之js插件
2015/11/30 Javascript
javascript动态添加checkbox复选框的方法
2015/12/23 Javascript
基于JS2Image实现圣诞树代码
2015/12/24 Javascript
jquery中ajax处理跨域的三大方式
2016/01/05 Javascript
JavaScript中各种引用类型的常用操作方法小结
2016/05/05 Javascript
jQuery内容过滤选择器用法示例
2016/09/09 Javascript
KnockoutJS 3.X API 第四章之数据控制流if绑定和ifnot绑定
2016/10/10 Javascript
JS DOMReady事件的六种实现方法总结
2016/11/23 Javascript
Bootstrap CSS组件之分页(pagination)和翻页(pager)
2016/12/17 Javascript
vue-property-decorator用法详解
2019/12/12 Javascript
NodeJS和浏览器中this关键字的不同之处
2021/03/03 NodeJs
wxPython学习之主框架实例
2014/09/28 Python
Python创建对称矩阵的方法示例【基于numpy模块】
2017/10/12 Python
Mac中Python 3环境下安装scrapy的方法教程
2017/10/26 Python
Python+logging输出到屏幕将log日志写入文件
2020/11/11 Python
联想法国官方网站:Lenovo法国
2018/10/18 全球购物
Nobody Denim官网:购买高级女士牛仔裤
2021/03/15 全球购物
安全负责人任命书
2014/06/06 职场文书
委托书的格式
2014/08/01 职场文书
代领毕业证委托书
2014/08/02 职场文书
优秀小学生事迹材料
2014/12/26 职场文书
电影开国大典观后感
2015/06/04 职场文书
springboot应用服务启动事件的监听实现
2022/04/06 Java/Android
Golang 实现WebSockets
2022/04/24 Golang