【问题标题】:Overwriting save method not working in Django Rest Framework覆盖保存方法在 Django Rest Framework 中不起作用
【发布时间】:2021-07-30 08:12:20
【问题描述】:

我有一个模型(积分),它根据用户的购买来保存积分。我的做法是,在下订单时(调用 orderapi),通过订单实例传递一个信号,并根据金额计算点数,并使用 save 方法保存。

创建了 Alothugh Order 对象,我没有看到保存在数据库中的点。我不确定是什么问题。

我的模型:

class Order(models.Model):
    ORDER_STATUS = (
        ('To_Ship', 'To Ship',),
        ('Shipped', 'Shipped',),
        ('Delivered', 'Delivered',),
        ('Cancelled', 'Cancelled',),
    )
    user = models.ForeignKey(User, on_delete=models.CASCADE, blank=True)   
    order_status = models.CharField(max_length=50,choices=ORDER_STATUS,default='To_Ship')
    ordered_date = models.DateTimeField(auto_now_add=True)
    ordered = models.BooleanField(default=False)   

    @property
    def total_price(self):
        # abc = sum([_.price for _ in self.order_items.all()])
        # print(abc)
        return sum([_.price for _ in self.order_items.all()])

    def __str__(self):
        return self.user.email

    class Meta:
        verbose_name_plural = "Orders"
        ordering = ('-id',)

class OrderItem(models.Model):       
    orderItem_ID = models.CharField(max_length=12, editable=False, default=id_generator)
    order = models.ForeignKey(Order,on_delete=models.CASCADE, blank=True,null=True,related_name='order_items')
    item = models.ForeignKey(Product, on_delete=models.CASCADE,blank=True, null=True)
    order_variants = models.ForeignKey(Variants, on_delete=models.CASCADE,blank=True,null=True)
    quantity = models.IntegerField(default=1)    

    @property
    def price(self):
        total_item_price = self.quantity * self.order_variants.price
        # print(total_item_price)
        return total_item_price    

    

class Points(models.Model):
    order = models.OneToOneField(Order,on_delete=models.CASCADE,blank=True,null=True)
    points_gained = models.IntegerField(default=0)

    def collect_points(sender,instance,created,**kwargs):
        total_price = instance.total_price
        if created:
            if total_price <= 10000:
                abc = 0.01 * total_price
            else:
                abc = 0.75 * total_price
            return abc

    post_save.connect(collect_points,sender=Order)

    def save(self,*args,**kwargs):
        self.points_gained = self.collect_points()
        super(Points, self).save(*args, **kwargs)

我实际上在这里感到困惑。我们可以使用 instance.total_price 访问属性 total_price 吗?

我的序列化器:

class OrderSerializer(serializers.ModelSerializer):
    billing_details = BillingDetailsSerializer()
    order_items = OrderItemSerializer(many=True)
    user = serializers.PrimaryKeyRelatedField(read_only=True, default=serializers.CurrentUserDefault())
    #total_price = serializers.SerializerMethodField(source='get_total_price')
    class Meta:
        model = Order
        fields = ['id','user','ordered_date','order_status', 'ordered', 'order_items','total_price', 'billing_details']
        # depth = 1   

    def create(self, validated_data):
        user = self.context['request'].user
        if not user.is_seller:
            order_items = validated_data.pop('order_items')
            billing_details = validated_data.pop('billing_details')
            order = Order.objects.create(user=user,**validated_data)
            BillingDetails.objects.create(user=user,order=order,**billing_details)
            for order_items in order_items:
                OrderItem.objects.create(order=order,**order_items)
            order.save()
            return order
        else:
            raise serializers.ValidationError("This is not a customer account.Please login as customer.")

我的更新代码:

class Order(models.Model):
total_price = models.FloatField(blank=True,null=True)

    def final_price(self):      
        return  sum([_.price for _ in self.order_items.all()])

    def save(self, *args, **kwargs):
        self.total_price = self.final_price()
        super(Order, self).save(*args, **kwargs)

class Points(models.Model):
    order = models.OneToOneField(Order,on_delete=models.CASCADE,blank=True,null=True)
    points_gained = models.FloatField(default=0)

    def collect_points(sender,instance,created,**kwargs):
        total_price = instance.total_price
        print(total_price)
        if created:
            if total_price <= 10000:
                abc = 0.01 * total_price
            else:
                abc = 0.75 * total_price
        new_point = Points.objects.create(order=instance, points_gained=abc)

    post_save.connect(collect_points,sender=Order)

专注于这部分

Class Order(models.Model):
    total_price = models.FloatField(blank=True,null=True)

    def final_price(self):
        # abc = sum([_.price for _ in self.order_items.all()])
        # print(abc)
        return  sum([_.price for _ in self.order_items.all()])

    def save(self, *args, **kwargs):
        self.total_price = self.final_price()
        super(Order, self).save(*args, **kwargs)

【问题讨论】:

    标签: django api django-rest-framework save django-signals


    【解决方案1】:

    虽然您确实将它放在您的 Point 模型中,但它不会影响它。它链接到Order 模型。如果我将您的信号翻译成英文,它将是:保存Order 实例后,计算并返回abc 变量

    我们绝不会在此处保存Point 模型或与之交互。因此,即使您确实覆盖了 Point.save() 方法,您也不会根据您的信号调用它

    如果你想做的是创建Point实例:

    def collect_points(sender, instance, created, **kwargs):
        total_price = instance.total_price
        if created:
            if total_price <= 10000:
                abc = 0.01 * total_price
            else:
                abc = 0.75 * total_price
            # ---> Now we can create the point
            new_point = Points.objects.create(order=instance, points_gained=abc)
    
    post_save.connect(collect_points,sender=Order)
    
    # And no need to override the save() method
    

    所以现在,我们的信号意味着在保存Order 实例后,如果它已创建,则计算点并根据此数据创建Point 实例

    此外,还有一些额外的提示可以让您在处理信号时更轻松:

    • 创建一个signals.py 文件。长期管理最好将您的信号分组到特定文件中
    • 把你的信号代码放在那里
    • 你可以在你的函数上使用装饰器来使它们成为信号,比如@receiver(post_save, sender=Order)
    • 在您的apps.py 文件中,在您的应用程序类中,覆盖ready 方法以导入信号。 ready 方法在启动应用程序时自动触发。所以这就像“在启动时执行此操作”。在我们的例子中,我们将在启动时注册信号。片段示例:
    #apps.py
    
    from django.apps import AppConfig
    
    class SecurityConfig(AppConfig):
    
        name = "security"
        label = "security"
    
        def ready(self):
            """Imports signals on application start"""
    
            import security.signals
    

    编辑:使用类方法和独立信号

    # In models.py
    class Points(models.Model):
        order = models.OneToOneField(Order,on_delete=models.CASCADE,blank=True,null=True)
        points_gained = models.IntegerField(default=0)
        
        @classmethod
        def create_point_from_order(cls, order_instance):
            """Creates a Points instance from an order"""
            total_price = order_instance.total_price
            if total_price <= 10000:
                abc = 0.01 * total_price
            else:
                abc = 0.75 * total_price
            return cls.objects.create(order=order_instance, points_gained=abc)
    

    然后您可以在调用该方法的Order 模型上创建一个信号

    # In signals.py
    from django.db.models.signals import post_save
    from django.dispatch import receiver
    from .models import Order, Points
    
    
    @receiver(post_save, sender=Order)
    def create_point_for_order(sender, instance, created, **kwargs):
        """On Order creation, creates a matching Points instance"""
        if created:
            created_point = Points.create_point_from_order(instance)
    

    最后,我们通过确保在启动时调用 signals.py 文件来注册该信号

    # In apps.py
    
    from django.apps import AppConfig
    
    class YourAppNameConfig(AppConfig):
    
        name = "your.app.name"
        label = "your.app.name"
    
        def ready(self):
            import yourappname.signals
    

    这样:

    • 逻辑仍在模型中,在create_point_from_order方法中
    • 这使我们可以从信号中分离逻辑,并使其可重用
    • 然后我们只需注册一个Order 信号,其工作就是简单地调用该点创建方法

    【讨论】:

    • 你好@Jordan,def collect_points 应该在 Points 模型中,对吧?你能检查我更新的代码吗?我现在将 total_price 定义为字段而不是属性,但它总是在创建订单时保存 0。
    • signal 只是在创建/保存/销毁模型时执行自动操作的一种方式。在那个信号中,你可以做任何你想做的事情。在您的情况下,您想要一个由Order 触发的信号,并且操作是创建Point。让我用一个关于如何正确做到这一点的例子来更新我的答案
    • 我已经更新了我的答案,并为signals.pyapps.py 提供了更多详细信息。这个想法是:逻辑在模型中(create_point_for_order),信号在signals.py(并最终调用你的模型逻辑),信号在启动时注册,因为它们是在ready中导入的您的应用配置方法
    • 好吧,我想我找到了。在您的序列化程序中,您正在创建 Order 实例(并保存它),然后它具有 OrderedItems。因此,当您保存它时,它没有OrderedItems,并且它的total_price 设置为0。您需要在添加OrderedItems 后重新保存它,以便重新计算其full_price
    • 在我们的例子中,我们想从Order 创建一个新的Points。我们还没有任何积分。所以使用Points 类而不是Points 实例是有意义的。我们使用@classmethod 装饰器表示This method uses the class object, not the instance。现在,我们创建该方法的原因是因为我们想在调用Points.objects.create 之前执行一些逻辑。但是最后一行cls.objects.create 完全是Points.objects.create。因为我们在Points 类中,所以cls == Points
    【解决方案2】:

    这是一个明显的例子,信号不适合。两种模型都在您的控制之下,您已经覆盖了 Order save 方法。这项工作需要使用信号,它只会使事情复杂化。

    这是简单的方法:

    class Order(models.Model):
        total_price = models.FloatField(blank=True,null=True)
    
        def final_price(self):      
            return  sum([_.price for _ in self.order_items.all()])
    
        def save(self, force_insert=False, **kwargs):
            created = force_insert or not self.pk
            self.total_price = self.final_price()
            super(Order, self).save(force_insert=force_insert, **kwargs)
            if created:
                points = Points.calculate_points(self.total_price)
                Points.objects.create(order=self, points_gained=points)
    
    class Points(models.Model):
        order = models.OneToOneField(
            Order,on_delete=models.CASCADE,blank=True,null=True
        )
        points_gained = models.FloatField(default=0)
    
        @staticmethod
        def calculate_points(amount):
            return 0.01 * amount if amount <= 10000 else 0.75 * amount
    
    

    信号是如何工作的?

    信号是库将任意代码注入到它们控制的一段代码中的方式:

    # Somebody else's code, you cannot modify
    def greeting():
        print('Hello ')
        send_signal('after_print_hello')
        print('!')
    

    现在这段代码有一些方法可以注册 'after_print_hello' 信号,这被称为接收器:

    # Your code
    def on_after_print_hello():
        print('world')
    

    所以最终会发生这种情况:

    def greeting():
        print('Hello ')
        print('world') <-----------\
        print('!')                 |
                                   |
    def on_after_print_hello():    |
        print('world')  -----------/
    

    如果代码的两部分都是你的,那么在这里使用信号似乎是绝对无稽之谈。您可以将打印语句移动到您想要的位置。

    如果你去掉所有花哨的注册和接收器匹配,这正是 Django 信号正在发生的事情。这些post_save 信号仅在您想在模型保存后 为您的模型中添加额外步骤时才有用。如果您将它们用于您自己的模型,您只需将 print('world') 语句移动到不同的位置,并使用 Django 使用更复杂的 API 为您调用它。

    【讨论】:

    • 所以我们不能使用信号,当发送者实例被保存方法覆盖时?你是这么说的吗?你能检查我更新的代码吗?它有效,但价格为零,积分也是如此。另外,如果我改用属性,我会得到 total_price。
    • 我是说你不必这样做。它使事情复杂化。您更新的代码让您远离解决方案。
    • 好的。但是我们必须使用@staticmethod 吗?我从来没有使用过它。我们不能只调用函数名而不使用 satticmethod 吗??
    • 是的。不要害怕。它不会受伤。 See here 解释一下。
    • 非常感谢@Melvyn,但我真的很关心一件小事。我已将更新后的代码作为重点代码。我只是对为什么总价格没有被保存感到困惑。它被保存为零。如果我将它用作属性,它只会显示价值。我现在已经删除了 Point 模型中的信号部分。你能检查一下吗??
    猜你喜欢
    • 1970-01-01
    • 2020-02-02
    • 1970-01-01
    • 1970-01-01
    • 2015-09-25
    • 1970-01-01
    • 1970-01-01
    • 2019-05-18
    • 1970-01-01
    相关资源
    最近更新 更多