为什么Django的post_save信号给我pre_save数据?

9 投票
3 回答
5029 浏览
提问于 2025-04-15 13:20

我正在尝试将一个“信息”对象连接到多个“客户”(见下面的代码)。

当这个信息对象被更新时,我想给每个与这个信息相关的客户发送电子邮件。

但是,当我记录信号接收到的 sold_to 字段时,我总是得到更新前的数据。

我猜这可能是因为这是一个多对多关系(ManyToManyField),数据存储在一个单独的表中,但难道在所有关系更新后,post_save 信号不应该被调用吗?

有没有人有解决方案的建议?

class Customer
    name = models.CharField(max_length=200)
    category = models.ManyToManyField('Category',symmetrical=False)
    contact = models.EmailField()

class Information
    name = models.CharField(max_length=200)
    email = models.EmailField(max_length=200)
    mod_date = models.DateTimeField(auto_now=True)
    sold_to = models.ManyToManyField(Customer, null=True, blank=True)


def send_admin_email(sender, instance, signal, *args, **kwargs):
    from myapp import settings
    for cust in instance.sold_to.all():
        settings.debug(cust.name)

post_save.connect(send_admin_email, sender=Information)

补充:apollo13 在 #django 提醒我这一点: “相关项(被保存到多对多关系中的东西)并不是作为模型的保存方法的一部分被保存的,正如你所发现的。” - http://groups.google.com/group/django-users/msg/2b734c153537f970

但因为这是2006年7月9日的内容,我真的希望现在有解决方案。

3 个回答

2

我遇到了同样的问题,因为我的模型里有多对多(M2M)字段,所以也碰到了类似的情况。

在这种情况下,问题在于多对多字段的两个相关模型都需要保存,才能获得自动生成的ID。

我的解决办法是,我没有使用post_save信号,也没有用m2m_changed信号,而是用了ModelAdmin类定义中的log_addition和log_change方法。

在你自定义的ModelAdmin类里:

    class CustomModelAdmin(admin.ModelAdmin):
         def log_addition(self, request, object):
         """
         Log that an object has been successfully added.
         """
             super(CustomModelAdmin, self).log_addition(request, object)
             #call post_save callback here object created

         def log_change(self, request, object):
         """
         Log that an object has been successfully changed.
         """
             super(CustomModelAdmin, self).log_change(request, object)
             #call post_save callback here object changed

如果你想的话,也可以重写log_deletion()方法。

祝你重写顺利……

2

这是我的解决方案,基于上面提到的来自code.djangoproject.com的补丁。

我在models.py文件中添加了以下内容:

from django.db.models.signals import m2m_changed
m2m_changed.connect(send_admin_email, sender=Information)

还有这个send_admin_email函数:

def send_customer_email(sender, instance, action, model, field_name, reverse, objects, **kwargs):
    if ("add" == action):
        # do stuff
5

你遇到的问题在这里有一个公开的记录,点击这里可以查看。你可以关注这个记录,看看什么时候会在新版本中解决这个问题,或者你也可以试着使用里面提供的修复方法,看看能否解决你的问题。

撰写回答