防止 Django 1.4.3 中 post_save 的重复信号

Ken*_*nny 4 python django duplicates

我正在尝试编写一些代码,每次其中一个用户修改模型对象时都会发送一封电子邮件。目前,我正在努力让 models.py 中的方法之一接收 post_save 信号。我意识到众所周知的事实是 post_save 信号通常发送两次,因此,解决方法是利用dispatch_uid 参数。我已经这样做了,但由于某种奇怪的原因,我继续收到两个信号。这是我的应用程序的 model.py 文件中的代码。

from django.db import models
from django.db.models.signals import post_save

def send_email(sender, **kwargs):
      print "Signal sent." #just a placeholder

post_save.connect(send_email, dispatch_uid="unique_identifier")

class Library_Associates (models.Model):
      first_name = models.CharField(max_length = 200)
      last_name = models.CharField(max_length = 200)

  department_choices = (
        ('ENG', 'Engineering'),
        ('ART', 'Arts and Sciences'),
        ('AFM', 'Accounting and Financial Managment'),
        ('MAT', 'Mathematics'),
  )

  department = models.CharField(max_length = 3, choices = department_choices, default = 'ENG')

  pub_date = models.DateTimeField ('date published')

  def __unicode__(self):
        return self.first_name

  class Meta:
        verbose_name_plural = 'Library Associates'

class Info_Desk_Staff (models.Model):
      first_name = models.CharField(max_length=50)
      last_name = models.CharField(max_length=50)
      salary = models.IntegerField()
      hours_worked = models.IntegerField()

      def __unicode__(self):
            return self.first_name

      class Meta:
            verbose_name_plural = 'Info Desk Staff'
Run Code Online (Sandbox Code Playgroud)

我已经多次重新启动服务器,重置/删除应用程序的所有数据,但我仍然收到两个信号。我的代码有本质上的错误吗?任何建议或见解将不胜感激!谢谢!

zif*_*fot 6

您的问题来自这样的事实:每次您通过管理界面修改对象时,管理应用程序都会创建代表所做更改的django.contrib.admin.models.LogEntry实例。

因为您正在侦听所有对象上的 post_save,所以您的侦听器会被调用两次 - 一次针对您的模型,第二次针对 LogEntry 模型。

可能的解决方案列表包括:

  1. 使用post_save方法中的sender参数为每​​个模型单独注册侦听器(例如以某种方式选择模型并在循环中执行)。

    for model in get_models():
        post_save.connect(send_email, sender = model, dispatch_uid='unique_identifier')
    
    Run Code Online (Sandbox Code Playgroud)
  2. 检查发送到监听器的发送者是否不是 django.contrib.admin.models.LogEntry 的实例

    from django.contrib.admin.models import LogEntry
    ...
    
    def send_email(sender, **kwargs):
        if isinstance(sender, LogEntry):
            return
    
    Run Code Online (Sandbox Code Playgroud)
  3. 为您的模型提供一个通用的超类,并使用它在侦听器中进行测试

    class MyModel(models.Model):
        pass
    
    class Library_Associates (MyModel):
        ...
    class Info_Desk_Staff (MyModel):
        ...
    
    def send_email(sender, **kwargs):
        if not isinstance(sender, MyModel):
            return
    
    Run Code Online (Sandbox Code Playgroud)