5

ドキュメントの1つの例を除けば、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のように、オブジェクト名に複数の単語が含まれている場合はどうなりますか?

補足:この方法でUserオブジェクトを拡張しようとしています。それが問題かもしれませんか?

4

1 に答える 1

13

カスタムを定義する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())

そして、これがそれをどのようにOneToOneField呼ぶかです:

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衝突に注意してください。

于 2010-09-13T23:32:20.450 に答える