Django post_save() signal implementation

匿名 (未验证) 提交于 2019-12-03 01:49:02

问题:

I have a question about django.

I have ManyToMany Models here

class Product(models.Model):      name = models.CharField(max_length=255)      price = models.DecimalField(default=0.0, max_digits=9, decimal_places=2)      stock = models.IntegerField(default=0)       def  __unicode__(self):          return self.name  class Cart(models.Model):     customer = models.ForeignKey(Customer)     products = models.ManyToManyField(Product, through='TransactionDetail')     t_date = models.DateField(default=datetime.now())     t_sum = models.FloatField(default=0.0)      def __unicode__(self):          return str(self.id)  class TransactionDetail(models.Model):     product = models.ForeignKey(Product)     cart = models.ForeignKey(Cart)     amount = models.IntegerField(default=0) 

For 1 cart object created, I can insert as many as new TransactionDetail object (the product and amount). My question is. How can I implement the trigger? What I want is whenever a Transaction detail is created, I want the amount of the product's stock is substracted by the amount in the transactiondetail.

I've read about post_save() but I'm not sure how to implement it. maybe something like this

when : post_save(TransactionDetail, Cart) #Cart object where TransactionDetail.cart= Cart.id

Cart.stock -= TransactionDetail.amount 

回答1:

If you really want to use signals to achieve this, here's briefly how,

from django.db.models.signals import post_save from django.dispatch import receiver  class TransactionDetail(models.Model):     # ... fields here  # method for updating @receiver(post_save, sender=TransactionDetail, dispatch_uid="update_stock_count") def update_stock(sender, instance, **kwargs):      instance.product.stock -= instance.amount      instance.product.save() 


回答2:

Personally I would override the TransactionDetail's save() method and in there save the new TransactionDetail and then run

self.product.stock -= self.amount self.product.save() 


回答3:

If you want to avoid getting maximum recursion depth exceeded, then you should disconnect signals, before saving within the signal handler. The example above (Kenny Shen's answer), would then be:

from django.db.models.signals import post_save from django.dispatch import receiver  class TransactionDetail(models.Model):     # ... fields here  # method for updating @receiver(post_save, sender=TransactionDetail, dispatch_uid="update_stock_count") def update_stock(sender, instance, **kwargs):  instance.product.stock -= instance.amount   post_save.disconnect(update_stock, sender=TransactionDetail)  instance.product.save()  post_save.connect(update_stock, sender=TransactionDetail) 

This is described thoroughly in Disconnect signals for models and reconnect in django, with a more abstract and useful example.

Also see: https://docs.djangoproject.com/en/2.0/topics/signals/#disconnecting-signals in the django docs.



标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!