保存时对相关模型执行操作

Posted

技术标签:

【中文标题】保存时对相关模型执行操作【英文标题】:Performing actions on a related model while saving 【发布时间】:2021-06-14 17:59:42 【问题描述】:

我需要帮助。几天来,我一直无法弄清楚如何解决一个问题。我将不胜感激任何想法或建议。 为简单起见,有一组模型:

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

事先不知道属于A的B的实例数。 B 的实例数由用户在创建新的 A 实例时确定。此外,他可以创建 A 的实例而无需创建 B 的单个实例(空集)。

挑战:我需要在保存 A 的新实例后运行 foo。此函数应同时处理 A 的新实例的字段和 B 的字段(反向关系)。

是的,我已将接收器设置post_save 信号模型 A。但问题是,此时 A 已被保存(post_save em>),B的实例还没有被保存(pre_save),对应字段的值还没有确定。换句话说,我无法在接收器 A 中获取原始 B 值。

有什么想法吗?我做错了吗?

【问题讨论】:

如果您需要在某些其他功能发生时发生特定功能,则需要在应用程序中实现信号。为此,您特别需要 post_save 方法。这是一个链接,可以了解有关信号如何工作的更多信息docs.djangoproject.com/en/3.1/topics/signals 还告诉我们有关函数 foo 的更多信息,以便我们提供解决方案。 感谢您的回答!当然,我已经实现了 post_save 信号接收器。但问题是,在信号到达的时候,还没有 B 的实例(A.b.None)实际上,这个函数应该动态创建一个名为 A.name 和字段集 B 的新模型 【参考方案1】:

我找到了解决方案。也许它对某人有用。 我听取了我们更有经验的同事的建议,将信号仅作为最后的手段。结果证明解决方案很简单。虽然解决方案可能看起来不是很优雅。

解决方案是基于覆盖每个模型的 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, *args, **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, *args, **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.owner._call_foo(self, pk, changed_fields)

此解决方案比使用信号更好,至少在真正需要时调用 save 方法:要么创建实例,要么更新它。

当然,需要记住的是,如果用户更新或创建了两个模型的多个实例(例如,在内联表单中,管理员创建了多个 B 实例和一个 A 实例),那么 foo( ) 函数将被调用多次。但这已经可以在 _call_foo() 方法级别解决,也可以通过 foo() 函数本身的逻辑来解决。

【讨论】:

以上是关于保存时对相关模型执行操作的主要内容,如果未能解决你的问题,请参考以下文章

使用 Objective-C 快速枚举时对 NSManagedContext 对象执行保存操作是不是安全

CakePHP- 保存相关模型数据

相关领域的 Django 模型验证

check-sat 返回未知时对部分模型的保证

Django REST框架:在ModelViewSet中保存相关模型

一.内存模型的相关概念