我有一个案例,用户需要更新一个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
来更新。
答案 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最长的操作离开发送电子邮件的地方。