Django Rest Framework更新嵌套的m2m对象。有谁知道更好的方法?

时间:2018-02-15 10:53:41

标签: python django django-rest-framework django-1.11

我有一个案例,用户需要更新一个instance以及在此实例上添加/编辑m2m相关对象。

这是我的解决方案:

# models.py

class AdditionalAction(SoftDeletionModel):
    ADDITIONAL_CHOICES = (
        ('to_bring', 'To bring'),
        ('to_prepare', 'To prepare'),
    )
    title = models.CharField(max_length=50)
    type = models.CharField(choices=ADDITIONAL_CHOICES, max_length=30)

class Event(models.Model):
     title= models.CharField(max_length=255)
     actions = models.ManyToManyField(AdditionalAction, blank=True)



# serializers.py
class MySerializer(serializers.ModelSerializer):
    def update(self, instance, validated_data):
        actions_data = validated_data.pop('actions')

        # Use atomic block to rollback if anything raised Exception 
        with transaction.atomic():

            # update main object
            updated_instance = super().update(instance, validated_data)

            actions = []
            # Loop over m2m relation data and
            # create/update each action instance based on id present
            for action_data in actions_data:
                action_kwargs = {
                    'data': action_data
                }
                id = action_data.get('id', False)
                if id:
                    action_kwargs['instance'] = AdditionalAction.objects.get(id=id)
                actions_ser = ActionSerializerWrite(**action_kwargs)
                actions_ser.is_valid(raise_exception=True)
                actions.append(actions_ser.save())

            updated_instance.actions.set(actions)

        return updated_instance

有人可以建议更好的解决方案吗?

P.S。在这种情况下可以创建或更新操作,因此我不能在序列化程序上使用many=True因为它还需要instance来更新。

1 个答案:

答案 0 :(得分:1)

如果你有一个很长的列表或在保存时触发的动作,那么在这里使用for循环将是一个杀手。我会尽量避免它。 最好使用ORMS更新和where子句:https://docs.djangoproject.com/en/2.0/topics/db/queries/#updating-multiple-objects-at-once,甚至在写入后从数据库中读取更新的对象。

要创建新操作,您可以使用bulk_create:https://docs.djangoproject.com/en/2.0/ref/models/querysets/#bulk-create

还有一个:https://github.com/aykut/django-bulk-update(免责声明:我不是该包的撰稿人或作者)。

您必须了解此方法的缺点 - 如果您使用任何post / pre_ save信号,那么更新不会触发这些信号。

通常,运行多次保存会终止数据库,最终可能难以诊断死锁。在其中一个项目中,我致力于从循环中的save()转换到update(),将响应时间从30秒减少到< 10最长的操作离开发送电子邮件的地方。