2011-04-07 29 views
26
ben daha iyi performans için bir veri denormalization yapmak ve benim blog yazısı Mesaj modeli içinde aldığı oy toplamı koymak istiyorum

orijinal (değiştirilmemiş) örneğini erişme:Django: post_save sinyal

class Post(models.Model): 
    """ Blog entry """ 
    author   = models.ForeignKey(User) 
    title   = models.CharField(max_length=255) 
    text   = models.TextField() 
    rating   = models.IntegerField(default=0) # here is the sum of votes! 

class Vote(models.Model): 
    """ Vote for blog entry """ 
    post   = models.ForeignKey(Post) 
    voter   = models.ForeignKey(User) 
    value   = models.IntegerField() 

Tabii, Post.rating değerini gerçek tutmam gerekiyor. Nornally Bunun için tetikleyici veritabanı kullanmak istiyorsunuz, ama şimdi bir post_save sinyalini yapmaya karar verdik (veritabanı işlemi süresini azaltmak için):

# vote was saved 
@receiver(post_save, sender=Vote) 
def update_post_votes(sender, instance, created, **kwargs): 
    """ Update post rating """ 
    if created: 
     instance.post.rating += instance.value 
     instance.post.save() 
    else: 
     # if vote was updated, we need to remove the old vote value and add the new one 
     # but how...? 

nasıl örnek değerini erişebilir kaydedildiği önce? Veritabanı tetikleyicilerinde, bunun için OLD ve NEW ön tanımları olurdu, ancak post_save sinyallerinde böyle bir şey var mı?

GÜNCELLEME

Mark'ın Yanıta göre çözüm:

# vote was saved 
@receiver(pre_save, sender=Vote) 
def update_post_votes_on_save(sender, instance, **kwargs): 
    """ Update post rating """ 
    # if vote is being updated, then we must remove previous value first 
    if instance.id: 
     old_vote = Vote.objects.get(pk=instance.id) 
     instance.post.rating -= old_vote.value 
    # now adding the new vote 
    instance.post.rating += instance.value 
    instance.post.save() 

cevap

38

Ben post_save değiştirilmemiş sürümünü almak için çok geç olduğuna inanıyoruz. Adından da anlaşılacağı gibi, veriler o noktada zaten db'ye yazılmıştır. Bunun yerine pre_save kullanmalısınız. Bu durumda, modeli db'den pk: old = Vote.objects.get(pk=instance.pk) üzerinden alabilir ve geçerli örnek ile önceki örnekte farklılıkları kontrol edebilirsiniz.

+0

Teşekkür ederim, kesinlikle haklısınız. Anlaşılır ki, instans yaratıldığında, pre_save sinyalinde 'id 'değeri yoktur, fakat eğer güncelleme yapıyorsa, bir id idrarına sahiptir. Tam ihtiyacım olan şey, gerçek çözüm için soru güncellemesini gör. –