Django当DEBUG = False时,信号不触发问题

今天突然发现生产环境中,某些信号没有被正确触发,一番调查总结一下关于django信号执行问题。

先看一个现象:

python 复制代码
# host/signals.py
@receiver(m2m_changed, sender=Vm.products.through)
def on_m2m_changed(sender, instance, action, pk_set, **kwargs):
    do something


@receiver(m2m_changed, sender=Vm.tags.through)
def on_m2m_changed(sender, instance, action, pk_set, **kwargs):
    do something


DEBUG=True时:
当修改Vm.products.through表时,第一个信号可以正常被触发,没什么问题
当修改Vm.tags.through表时,第二个信号可以正常被触发,也没什么问题
DEBUG=False时:
当修改Vm.products.through表时,第一个信号不能被正确触发
当修改Vm.tags.through表时,第二个信号可以正常被触发,没什么问题

也就是说是否开启debug模式会影响第一个信号的触发,查了下官方文档发现:

警告

还需要注意的是,Django 默认将信号处理程序存储为弱引用,所以如果你的处理程序是一个本地函数,它可能会被垃圾回收。 为了防止这种情况,当你调用信号的 connect() 时,传入 weak=False

然后尝试传入weak=False,果然信号被正确触发了。看来真的如文档所说的一样,第一个信号处理函数被垃圾回收了。

那不对啊,项目中其他用到信号的地方也没加weak参数,也没发现这种信号不被触发的问题。

回头看看,发现我这两个多对多信号处理函数名称相同了,尝试换了不一样的函数名,即使不使用weak参数,两个信号处理函数也能被正确触发。

至此问题已清晰明了,是因为两个信号处理函数同名了,导致第一个被垃圾回收了。

总结:

同一个文件中的信号处理函数名称不能一样,否则靠前的函数会被垃圾回收,除非传入weak=False参数声明为强引用。

相关推荐
rannn_1112 分钟前
【Redis|高级篇2】多级缓存|JVM进程缓存、Lua语法、多级缓存实现(OpenResty)、缓存同步(Canal)
java·redis·分布式·后端·缓存·lua·openresty
Allen_LVyingbo7 分钟前
《狄拉克符号法50讲》习题与解析(上)
开发语言·人工智能·python·数学建模·量子计算
betazhou8 分钟前
django+postgresql创建第一个网站
python·postgresql·django
AC赳赳老秦8 分钟前
OpenClaw对接百度指数:关键词热度分析,精准定位博客创作方向
java·python·算法·百度·dubbo·deepseek·openclaw
Ava的硅谷新视界10 分钟前
SQLite WAL 模式踩坑笔记:高并发读写下的几个细节
开发语言·后端·编程
u01074754611 分钟前
SQL在分布式SQL环境下的JOIN性能优化_减少跨节点数据传输
jvm·数据库·python
雨墨✘14 分钟前
CSS如何实现不同屏幕下的字体缩放_利用clamp函数动态调整
jvm·数据库·python
hef28817 分钟前
Go语言如何刷LeetCode_Go语言LeetCode刷题教程【速学】
jvm·数据库·python
u01074754619 分钟前
HTML5中SVG描边虚线Stroke-dasharray的配置技巧
jvm·数据库·python
万世浮华戏骨21 分钟前
Web 后端 Python 基础安全
前端·python·安全