问题描述
假设我有一个名为“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'
解决方法
所以我认为您在这里对 parent
和 child
的命名感到困惑。 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