Django - 多态模型还是一个大模型?

Posted

技术标签:

【中文标题】Django - 多态模型还是一个大模型?【英文标题】:Django - Polymorphic Models or one big Model? 【发布时间】:2013-04-25 00:35:51 【问题描述】:

我目前正在 Django 中研究一个模型,该模型涉及一个模型,该模型可以具有各种不同的特征,具体取决于它是什么类型的对象。因此,假设我们有一个名为 Mammal 的模型,它可以是大象或海豚(分别具有自己的特征“tusk_length”和“flipper_length”)。

基本的 OOP 原则大喊“多态性”,我倾向于同意。但是,由于我是 Django 新手,我首先想知道这是否是 Django 中最好的方法。我听说过很多例子,有些人对奇异的巨型模型有自己的偏好

我已经尝试过使用 GenericForeignKeys,如下所述:How can I restrict Django's GenericForeignKey to a list of models?。虽然这个解决方案效果很好,但我不喜欢无法过滤,而且这种关系只是一种方式。也就是说,虽然您可以从 Mammal 对象中获取 Dolphin,但无法从 Dolphin 中获取 Mammal 对象。

所以,这是我的两个选择:

选择 A:

from django.db import models
class Mammal(models.Model):
    hair_length = models.IntegerField()
    tusk_length = models.IntegerField()
    flipper_length = models.IntegerField()
    animal_type = models.CharField(max_length = 15, choices = ["Elephant", "Dolphin"]

选择 B:

from django.db import models
class Mammal(models.Model):
    hair_length = models.IntegerField()

class Elephant(Mammal):
    tusk_length = models.IntegerField()

class Dolphin(Mammal):
    flipper_length = models.IntegerField()

据我了解,选项 B 在查询和列出所有大象或海豚时具有更好的代码优势。但是,我注意到在不将 animal_type 放入类中的情况下,从哺乳动物列表中获取所有大象并不是那么简单(是否有对此的查询?),默认情况下依赖于类。

这导致我看到的另一个多态性问题,在上面的示例或我的应用程序中不会出现,但值得一提的是,如果不完全删除 Dolphin,就很难将 Dolphin 对象编辑为 Elephant .

总的来说,有什么普遍的偏好,或者我不应该使用多态性的任何重要原因吗?

【问题讨论】:

【参考方案1】:

我们最终选择了一个包含许多通常为空的列的大表。我们的理由是(在这种情况下)我们的 Mammal 表就是我们要查询的全部内容,除了手动检查它们是否有“海豚”或“大象”之外,没有(直观的)方法可以过滤某些类型的哺乳动物" 对象,如果没有,则会引发错误。即使寻找从绝对是大象的查询返回的对象的类型,仍然返回“哺乳动物”。很难将任何 Pythonic 变通方法扩展到编写纯 SQL,而我们的一位数据专家经常这样做。

【讨论】:

AFAIK 您的解决方案称为“单表继承”。【参考方案2】:

对于数据库设计,我的建议是避免继承。它使访问和更新都变得复杂。

在 Django 中,尝试为您的基础模型使用抽象类。这意味着不会为其创建数据库表。它的字段/列将在其子模型中自动创建。好处是:Django/Python 代码中的代码重用和数据库中的简单、扁平化设计。惩罚是:管理/查询子模型的混合集合需要更多的工作。

在此处查看示例:Django Patterns: Model Inheritance

或者,您可以将“哺乳动物”的概念更改为“哺乳动物特征”。并在每个特定的哺乳动物类中包含一个 MammalTraits 对象。在代码中,这就是组合(has-a)。在数据库中,这将表示为外键。

【讨论】:

现在,我看到使用抽象类的一个主要问题(在我的例子中,不是一般情况下)是生产中已经存在并填充了 Mammal 表。将其更改为“MammalTraits”可能会起作用,但我想在一个 MammalTrait 和一个 Dolphin/Elephant 对象之间强制建立一对一的关系。这在理论上当然很简单,但我必须强制只有一只海豚或一只大象,我认为这并不太难(覆盖 save() 方法进行验证?)。 嗯。不过,我确实看到了您对多个子模型的看法。按海豚过滤所有哺乳动物会有点麻烦...... 重新审视了这一点——实际上我最终使用了一个抽象类作为我正在做的新事物的基础模型。所以对我来说,结论是使用你在这里建议的两件事之一来创建一个新模型,但如果它是一个现有模型,需要与现有的实时数据变得多态,那么用你的所有字段饱和它可能会更安全需要。

以上是关于Django - 多态模型还是一个大模型?的主要内容,如果未能解决你的问题,请参考以下文章

Django 外键关系应该是用户到模型还是模型到用户?

如何在 django 中进行复杂的搜索?使用视图还是模型?

解析 Django 模型服务器端还是客户端?

我的模型文件应该保留在不同的包中还是 Django 中的相同模型文件中?

在 Django 中扩展用户对象:用户模型继承还是使用 UserProfile?

Django数据存储 - SQL还是别的什么?