3

这让我感到困惑......当我保存我的模型时,书籍对象没有改变。但是,如果我打开发票并再次保存,就会进行更改。我究竟做错了什么?

class Invoice(models.Model):
    ...
    books = models.ManyToManyField(Book,blank=True,null=True)
    ...

    def save(self, *args, **kwargs):
        super(Invoice, self).save(*args, **kwargs)
        for book in self.books.all():
            book.quantity -= 1
            if book.quantity == 0:
                book.sold = True;
            book.save()

编辑:我尝试过使用 post_save 信号,但它的工作方式相同。第一次保存没有更改,第二次保存更改。

更新:似乎可以用这段代码解决:

class InvoiceAdmin(admin.ModelAdmin):
    ...

    def save_model(self, request, obj, form, change):
        obj.save()
        for bk in form.cleaned_data['books']:
            book = Book.objects.get(pk=bk.id)
            book.quantity -= 1
            if book.quantity == 0:
                book.sold = True;
            book.save()
4

2 回答 2

5

这就是我解决这个确实令人困惑的行为的方法。将信号接收器连接到 models.signals.m2m_changed 事件,每次更改 m2m 字段时都会触发此获取。内联注释解释了原因。

class Gig(models.Model):
    def slugify(self):
        # Add venue name, date and artists to slug
        self.slug = slugify(self.venue.name) + "-"
        self.slug += self.date.strftime("%d-%m-%Y") + "-"
        self.slug += "-".join([slugify(artist.name) for artist in self.artists.all()]) 
        self.save()


@receiver(models.signals.m2m_changed, sender=Gig.artist.through)
def gig_artists_changed(sender, instance, **kwargs):
    # This callback function get's called twice. 
    # 1 first change appears to be adding an empty list
    # 2nd change is adding the actual artists
    if instance.artist.all() and not instance.slug:                                                                                                                                                               
        instance.slugify()
于 2012-08-17T19:25:02.843 回答
1

那是因为m2m关系是在你的模型保存后保存的,以便获得父对象的PK。在您的情况下,第二次保存按预期工作,因为模型已经具有第一次保存的 PK 和相关书籍(它在信号中完成)。

我还没有找到解决方案,最好的办法是在管理视图中进行更改,我猜。

于 2010-02-04T08:14:17.350 回答