Django的一对一关系如何将名称映射到子对象?
除了文档中的一个例子,我找不到关于Django是如何选择从父对象访问子对象的名称的具体说明。在他们的例子中,他们做了以下操作:
class Place(models.Model):
name = models.CharField(max_length=50)
address = models.CharField(max_length=80)
def __unicode__(self):
return u"%s the place" % self.name
class Restaurant(models.Model):
place = models.OneToOneField(Place, primary_key=True)
serves_hot_dogs = models.BooleanField()
serves_pizza = models.BooleanField()
def __unicode__(self):
return u"%s the restaurant" % self.place.name
# Create a couple of Places.
>>> p1 = Place(name='Demon Dogs', address='944 W. Fullerton')
>>> p1.save()
>>> p2 = Place(name='Ace Hardware', address='1013 N. Ashland')
>>> p2.save()
# Create a Restaurant. Pass the ID of the "parent" object as this object's ID.
>>> r = Restaurant(place=p1, serves_hot_dogs=True, serves_pizza=False)
>>> r.save()
# A Restaurant can access its place.
>>> r.place
<Place: Demon Dogs the place>
# A Place can access its restaurant, if available.
>>> p1.restaurant
所以在他们的例子里,他们直接使用p1.restaurant来访问,而没有明确指定这个名称。Django默认这个名称是小写开头的。如果对象的名称有多个单词,比如FancyRestaurant,那会怎么样呢?
顺便提一下:我正在尝试以这种方式扩展用户对象。这可能是问题所在吗?
1 个回答
15
如果你定义了一个自定义的 related_name
,那么系统会使用你定义的这个名字。否则,它会把整个模型的名字变成小写(在你的例子中是 .fancyrestaurant
)。你可以查看 django.db.models.related 代码 中的 else 部分来了解更多:
def get_accessor_name(self):
# This method encapsulates the logic that decides what name to give an
# accessor descriptor that retrieves related many-to-one or
# many-to-many objects. It uses the lower-cased object_name + "_set",
# but this can be overridden with the "related_name" option.
if self.field.rel.multiple:
# If this is a symmetrical m2m relation on self, there is no reverse accessor.
if getattr(self.field.rel, 'symmetrical', False) and self.model == self.parent_model:
return None
return self.field.rel.related_name or (self.opts.object_name.lower() + '_set')
else:
return self.field.rel.related_name or (self.opts.object_name.lower())
class OneToOneField(ForeignKey):
... snip ...
def contribute_to_related_class(self, cls, related):
setattr(cls, related.get_accessor_name(),
SingleRelatedObjectDescriptor(related))
opts.object_name(在 django.db.models.related.get_accessor_name 中提到) 默认是 cls.__name__
。
至于
附带说明:我试图以这种方式扩展 User 对象。这可能是问题所在吗?
不是的,User
模型只是一个普通的 django 模型。只要注意 related_name
的冲突就可以了。