是否可以为外键关系添加父级到子级?

问题描述

假设我有一个名为“parent”的 ForeignKey 字段,其相关名称为“children”:

class Item(polymorphicModel):
    title = models.CharField()
    parent = models.ForeignKey(
        "self",related_name='children',null=True,blank=True,on_delete=models.CASCADE)

class Parent(Item):
    pass

class Child(Item):
    pass

为什么我只能将子项添加到父项,但尝试将父项添加到子项时会出现错误

所以这是有效的:

p1 = Parent.objects.create(title="Parent 1")
c1 = Child.objects.create(title="Child 1")

print(p1.children)
#<polymorphicQuerySet []>

p1.children.add(c1)

但这不会:

p1 = Parent.objects.create(title="Parent 1")
c1 = Child.objects.create(title="Child 1")
print(c1.parent)
# None

c1.parent.add(p1)
# AttributeError: 'nonetype' object has no attribute 'add'

我是否只需要每次都添加到 Parent 的 children 字段中?有没有办法添加到孩子的父母?是否有任何理由为什么添加到孩子的父级不起作用或不应该使用?

我也对在这种情况下何时/如何使用“_set”(如果相关)感到有些困惑。因此,按照 Django's Many-to-one example 的格式,以下对我也不起作用:

p1 = Parent.objects.create(title="Parent 1")
c1 = Child.objects.create(title="Child 1")
p1.children.add(c1)

print(p1.children_set.all())
# AttributeError: 'p1' object has no attribute 'children_set'

print(c1.parent_set.all())
# AttributeError: 'c1' object has no attribute 'parent_set'

print(p1.item_set.all())
# AttributeError: 'p1' object has no attribute 'item_set'

解决方法

所以我认为您在这里对 parentchild 的命名感到困惑。 related_name 中的 ForeignKey 字段本质上是告诉模型建立关系以快速找到具有给定 ForeignKey 的模型的所有相关实例。当您调用 p1.children 时,您会得到正确的输出,因为没有与 p1 相关的实例。当您调用 c1.parent 并获得 None 时,您同样会得到正确的输出。我在下面复制和粘贴的行是导致这种情况的原因。通过设置 null=True,您是说要实例化 Item 实例(无论是 Parent 还是 Child),并带有一个 null parent 字段(在 python 中,null是 None)。

当您拨打 c1.parent.add() 时,您感觉到的问题就出现了。由于您尚未将 parent 设置为任何内容,因此其值为 None,并且它同样没有 add() 方法。您应该做的是设置 parent=[some instance of Parent]。然后,当您想获取给定 children 实例的 Parent 时,比如说 p1,您可以调用 p1.children,您将获得一个充满 {{ 1}} 个实例,其 Child 字段已设置为该 parent 实例的外键。

Parent

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...