如何解决保存时对相关模型执行操作
我需要帮助。几天来,我一直无法弄清楚如何解决一个问题。我将不胜感激任何想法或建议。 为简单起见,有一组模型:
class A(models.Model):
name = models.CharField(max_length=255)
class B(models.Model):
name = models.CharField(max_length=255)
owner = models.ForeignKey(A,on_delete=models.CASCADE,null=False)
def foo():
do something
B 属于 A 的实例数量是事先未知的。 B 的实例数由用户在创建 A 的新实例时决定。此外,他可以创建 A 的实例,而无需创建 B 的单个实例(空集)。
挑战:我需要在保存 A 的新实例后运行 foo。这个函数应该处理 A 的新实例的字段和 B 的字段(反向关系)。
是的,我已将接收器设置为post_save信号模型 A。但问题来了,此时 A 已经被保存 (post_save em>),B的实例还没有保存(pre_save),对应字段的值还没有确定。换句话说,我无法在接收者 A 中获取原始 B 值。
有什么想法吗?我做错了什么吗?
解决方法
我找到了解决方案。也许它会对某人有用。 我听取了我们更有经验的同事的建议,只将信号作为最后的手段。结果证明解决方案很简单。虽然解决方案可能看起来不太优雅。
解决方案基于覆盖每个模型的save()方法
class A(models.Model):
name = models.CharField(max_length=255)
def _call_foo(self,*args,**kwargs):
do something
foo(*args,**kwargs)
def save(self,**kwargs):
changed_fields = []
pk = None
if self.pk:
cls = self.__class__
old = cls.objects.get(pk=self.pk)
new = self
for field in cls._meta.get_fields():
field_name = field.name
try:
if getattr(old,field_name) != getattr(new,field_name):
pk = self.pk
changed_fields.append(field_name)
# Catch field does not exist exception
except Exception as ex:
print(type(ex))
print(ex.arg)
print(ex)
kwargs['update_fields'] = changed_fields
super().save(*args,**kwargs)
else:
super().save(*args,**kwargs)
pk = self.pk
changed_fields.append('__all__')
self._call_foo(self,pk,changed_fields)
class B(models.Model):
name = models.CharField(max_length=255)
owner = models.ForeignKey(A,on_delete=models.CASCADE,null=False)
def save(self,**kwargs)
pk = self.pk
changed_fields.append('__all__')
self.owner._call_foo(self,changed_fields)
此解决方案比使用信号更好,至少在确实需要时调用 save 方法:要么创建实例,要么更新实例。
当然,需要记住的是,如果用户更新或创建了两个模型的多个实例(例如,在内联表单中,管理员创建了多个 B 的实例和一个 A 的实例),那么 foo( ) 函数将被多次调用。但这已经可以在 _call_foo() 方法级别或通过 foo() 函数本身的逻辑解决。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。