[TimLinux] Django 信号
1. 信号定义
django包含有一个“信号分发器”,在框架内任何时候,在任何地方,有动作发生时,用来帮助解耦应用之间获取通知。简言之,信号允许特定的发送者通知一系列接收者某一特定动作已经发生了。特别有用的地方在于:许多代码片段对于同样的事件感兴趣。
2. 内建信号
django提供了许多内建的信号,这样用户代码能够在django自身的一些特定动作发生时得到通知。如下:
- django.db.models.singals.pre_save、django.db.models.signals.post_save:在模型的save()方法被调用之前、之后发送该信号。
- django.db.models.signals.pre_delete、django.db.models.signals.post_delete:在模型的delete()方法被调用之前、之后发送该信号。
- django.db.models.signals.m2m_chagned:当一个ManyToManyField模型发送修改是发送该信号。
- django.core.signals.request_started、django.core.signals.request_finished:在django开始、完成一个HTTP请求时发送该信息。
- 等等其他信号。
3. 监听信号
为了接收一个信号,使用Signal.connect()方法来注册一个信号接收函数,当信号被发送的时候接收函数将被调用。函数接口如下:
Signal.connect(receiver, sender=None, weak=True, dispatch_uid=None) 参数说明:
receiver: 回调函数,将被连接到对象的信号上。receiver函数有它的实现要求
sender: 指定一个特定的发送者,明确信号从哪里来的。
weak: django存储信号处理器默认作为弱引用。可以通过设置weak=False来修改这一行为。
dispatch_uid: 信号接收者的唯一标识,以免重复信号被发送。 例如:request_finished就是一个Signal对象,下面这行代码完成了信号的监听
request_finished.connect(my_callback) 下面的函数定义,完成了接收函数的定义
def my_callback(sender, **kwargs):
print("Request finished!")
3.1. 接收函数
接收函数可以是任意Python函数或方法:接收一个sender位置参数,一个通配的kwargs关键字参数,所有信号处理器必须接收这些参数。所有的信号发生kwargs参数,并且可能在任一时间改变这些关键字参数。根据request_finished信号文档说明,它不会发送任何参数,看似上面的回调函数也可能不传递关键字参数,但是这样的操作是错误的,如果你真的这样做,django会抛出一个错误的。
3.2. 连接接收函数
两种方式可以将一个接收者连接到信号上去。
# 1. 手动连接
from django.core.signals import request_finished def my_callback(sender, **kwargs):
print("Request finished!")
request_finished.connect(my_callback) # 2. 使用 receiver 装饰器连接
from django.core.signals import request_finished
from django.dispatch import receiver @receiver(request_finished)
def my_callback(sender, **kwargs):
print("Request finished!")
现在my_callback函数会在每次完成完成一次请求之后被调用。
3.3. 连接信号:指定发送者
指定发送者的目的是缩小事件的监听范围,比如pre_save()信号,多数情况下只是想知道某一个特定的模型上发生的这一信号,而不是所有的模型。这时候就可以指定你所关心的sender,比如sender=User,那么只有当User模型在pre_save信号发送时,才调用receiver函数。不同的信号,能够接收的sender参数是值不同,需要根据信号文档描述进行设置,不能乱猜哦!
from django.core.signals import pre_save
from django.dispatch import receiver
from myapp.models import User @receiver(pre_save, sender=User)
def my_handler(sender, **kwargs):
print("pre_save happen in User.")
3.4. 阻止重复信号
在一些特定的情形下,连接在信号上的接收函数可能被运行多次,这可能导致你的接收函数被注册多次,并且这也就导致了同一信号事件发生时接收函数被调用多次。如果这样的行为导致问题(比如:当一个模型存在save操作是,使用信号来发送邮件通知),传递一个唯一标识符来来标识你的接收函数避免这样的情况(dispatch_uid),这个标识符通常使用字符串(其实可以使用任一可hash的对象)。最终结果是信号接收函数只会绑定一次。
from django.core.signals import request_finished request_finished.connect(my_callback, dispatch_uid="my_unique_identifier")
4. 定义和发送信号
4.1. 定义信号
所有信号都是django.dispatch.Signal的实例,providing_args是一个由参数名组成的列表,将由信号提供给监听者。
import django.dispatch pizza_done = django.dispatch.Signal(providing_args=["toppings", "size"]) 声明了pizza_done信号,将提供toppings、size两个参数给接收者。
注:你可以在任何时候修改这个providing_args列表,所有没必要在第一个确保参数的正确。
4.2. 发送信号
两种方式发送信号:
- Signal.send(sender, **kwargs): 所有内建信号都是采用这个方式来发送的。
- Signal.send_robust(sender, **kwargs):
两种方式都需要一个sender作为参数(多数时候是一个类),并且可能提供其他你所想要的关键字参数。如下发送描述怎样发送一个pizza_done信号:
class PizzaStore(object):
...
def send_pizze(self, toppings, size):
pizza_done.send(sender=self.__class__, toppings=toppings, size=size)
...
send、send_robust都返回一个元祖对构成的列表:[ (receiver, response), ...],呈现一个调用了的接收函数和它们的response值。send比send_robust区别的地方在于由接收函数抛出的异常时如何被处理的。send不会捕捉任何由receiver函数抛出的异常,它简单的允许异常被冒泡传递。这也是当一个信号内部遇到错误时,不是所以的接收这都将被通知的原因。send_robust会使用python Exception这个比较顶层的异常类来捕捉所有的错误,并且确保所有的接收者都被通知到,错误被存放在__traceback__属性中。
5. 断开信号
Signal.disconnect(receiver=None, sender=None, dispatch_uid=None),参数与在Signal.connect()中描述的一样,返回:True断开成功,False断开失败。如果dispatch_uid被使用的时候,receiver可以传递None。
6. 内建信号
# 1. 模型信号
pre_init post_init pre_save post_save
pre_delete post_delete m2m_chagned class_prepared # 2. 管理信号
pre_migrate post_migrate # 3. 请求、响应信号
request_started request_finished get_request_exception # 4. 测试信号
setting_changed template_rendered # 5. 数据库包装器
connection_created
7. 信号注册地
可以放在AppConfig类中,AppConfig类提供了一个ready函数,这个函数的执行是由Django框架自动调用的,调用的时机是当所有的(INSTALLED_APPS)都被注册完成后。
# 介绍文档:API Reference#Applications.
from django.db.models.signals import pre_save class PollsConfig(AppConfig):
name = 'polls' def ready(self):
from .models import MyModel
# 或者:
# MyModel= self.get_model('MyModel')
pre_save.connect(receiver, sender='app_label.MyModel')
[TimLinux] Django 信号的更多相关文章
- Django 信号signal
序言 Django自带一套信号机制来帮助我们在框架的不同应用位置之间传递信息.也就是说,当某一事件发生时,信号系统可以允许一个或多个发送者(senders)将信号(signals)发送给一组接收者(r ...
- 信号(Django信号、Flask信号、Scrapy信号)
简介 Django.Flask.scrapy都包含了一个“信号分配器”,使得当一些动作在框架的其他地方发生的时候,解耦的应用可以得到提醒. 通俗来讲,就是一些动作发生的时候,信号允许特定的发送者去提醒 ...
- Django信号机制相关解释与示例
Django 信号# django自带一套信号机制来帮助我们在框架的不同位置之间传递信息.也就是说,当某一事件发生时,信号系统可以允许一个或多个发送者(senders)将通知或信号(signals)发 ...
- Django 信号使用问题
Django 信号使用问题: 在使用django内置信号修改新注册的用户密码的时候,发现内置信号没有被触发.百度&官方文档找到了答案 1.信号的函数应该放在哪里? 这段代码应该放在哪里? 严格 ...
- flask,scrapy,django信号
简介 Django.Flask.scrapy都包含了一个“信号分配器”,使得当一些动作在框架的其他地方发生的时候,解耦的应用可以得到提醒. 通俗来讲,就是一些动作发生的时候,信号允许特定的发送者去提醒 ...
- pythonのdjango 信号
一.内置信号 Django中提供了“信号调度”,用于在框架执行操作时解耦.通俗来讲,就是一些动作发生的时候,信号允许特定的发送者去提醒一些接受者. Model signals pre_init # d ...
- django信号浅谈
Django中提供了“信号调度”,用于在框架执行操作时解耦.通俗来讲,就是一些动作发生的时候,信号允许特定的发送者去提醒一些接受者. 1.Django内置信号 Model signals pre_in ...
- django信号
什么是信号? 信号是在某个操作前或后自动触发一些操作. 信号是通知,是一种状态,相当于在某种状态下发特定的消息 --为了实现代码层解耦 村长博客:http://www.cnblogs.com/legu ...
- Django 信号
信号 Django 提供一个“信号分发器”,允许解耦的应用在框架的其它地方发生操作时会被通知到. 简单来说,信号允许特定的sender通知一组receiver某些操作已经发生. 这在多处代码和同一事件 ...
随机推荐
- python机器学习——逻辑回归
我们知道感知器算法对于不能完全线性分割的数据是无能为力的,在这一篇将会介绍另一种非常有效的二分类模型--逻辑回归.在分类任务中,它被广泛使用 逻辑回归是一个分类模型,在实现之前我们先介绍几个概念: 几 ...
- 深入理解 PHP 的 7 个预定义接口
深入理解预定义接口 场景:平常工作中写的都是业务模块,很少会去实现这样的接口,但是在框架里面用的倒是很多. 1. Traversable(遍历)接口 该接口不能被类直接实现,如果直接写了一个普通类 ...
- 本地Git连接GitLab(服务器)远程仓库
1.简介 远程仓库是指托管在网络上的项目仓库,现在互联网上有很多项目托管平台,比如github.gitlab等.为了不公开自己项目代码,可以在自己的服务器上搭建自己的项目仓库,最常见的是搭建GitLa ...
- nyoj 76-超级台阶 (递推)
76-超级台阶 内存限制:64MB 时间限制:1000ms 特判: No 通过数:8 提交数:12 难度:3 题目描述: 有一楼梯共m级,刚开始时你在第一级,若每次只能跨上一级或二级,要走上第m级,共 ...
- centos安装后第一次重启,许可协议、Kdump
1.许可协议,服务器键盘操作找到许可 确定(遇到过,第一次懵逼了) 2.Kdump是RHEL提供的一个崩溃转储功能,用于在系统发生故障时提供分析数据,它会占用系统内存,一般选择关闭(默认是关闭)(这个 ...
- 菜鸟手把手学Shiro之shiro认证流程
一.使用的spring boot +mybatis-plus+shiro+maven来搭建项目框架 <!--shiro--> <dependency> <groupId& ...
- think PHP 查询、更改
最近公司没有什么新项目,故准备搞搞PHP,正好后端有一些小东西需要搞一下,我就来试试吧. PHP 基于think PHP 3 实现功能: 1.为销售绑定虚拟号码分组(查询可以绑定的分组 -> 绑 ...
- C博客作业00—我的第一篇博客
C博客作业00-我的第一篇博客 1. 你对网络专业或者计算机专业了解是怎样? 泛泛了解 - 原先只知道网络工程隶属于计算机工程学院,与院中其他专业一样,同样都需要学习大量的计算机基础知识,然后再分支学 ...
- 数位dp介绍
不了解dp的可以先看一下dp 数位dp含义: 数位:一个数有个位,十位,百位,千位等等,数的每一位都是数位. 数位dp归为计数dp,是在数位上进行操作的dp. 数位dp的实质是一种快速枚举的方式,它满 ...
- 【设计模式大法】Iterator模式
Iterator模式 --一个一个遍历 在Java中的for语句中 i++的作用是让 i 的值在每次循环后自增1,这样就可以访问数组中的下一个元素.下下一个元素.再下下一个元素,也就实现了从头至尾逐一 ...