Django:为什么外键查找不会自动使用pk?

时间:2009-07-16 09:22:25

标签: django django-models django-queryset

我有

class Achievement(MyBaseModel):
    pass

class Alias(MyBaseModel):
    achievements = models.ManyToManyField('Achievement')

>>> ach = Achievement.objects.all()[1]

这有效:

>>> Alias.objects.all().filter(achievements__pk__contains=ach.pk).count()
77L

但这不是:

>>> Alias.objects.all().filter(achievements__contains=ach).count()
Traceback (most recent call last):
  File "<console>", line 1, in <module>
  File "/var/home/ptarjan/django/mysite/django/db/models/query.py", line 489, in filter
    return self._filter_or_exclude(False, *args, **kwargs)
  File "/var/home/ptarjan/django/mysite/django/db/models/query.py", line 507, in _filter_or_exclude
    clone.query.add_q(Q(*args, **kwargs))
  File "/var/home/ptarjan/django/mysite/django/db/models/sql/query.py", line 1258, in add_q
    can_reuse=used_aliases)
  File "/var/home/ptarjan/django/mysite/django/db/models/sql/query.py", line 1201, in add_filter
    self.where.add((alias, col, field, lookup_type, value), connector)
  File "/var/home/ptarjan/django/mysite/django/db/models/sql/where.py", line 48, in add
    params = field.get_db_prep_lookup(lookup_type, value)
  File "/var/home/ptarjan/django/mysite/django/db/models/fields/related.py", line 156, in get_db_prep_lookup
    raise TypeError, "Related Field has invalid lookup: %s" % lookup_type
TypeError: Related Field has invalid lookup: contains

为什么呢? (Django 1.0.2)

查看查询日志,它正在做一些我没想到的事情!该查询产生了:

>>> connection.queries[-1]
{'time': '0.027', 'sql': u'SELECT COUNT(*) FROM `yourock_alias` INNER JOIN `yourock_achiever` ON (`yourock_alias`.`id` = `yourock_achiever`.`alias_id`) WHERE `yourock_achiever`.`achievement_id` LIKE BINARY %j0xvw9% '}

但是这样做

>>> Alias.objects.all().filter(achievements=ach).count()
77L

提供此查询

>>> connection.queries[-1]
{'time': '0.023', 'sql': u'SELECT COUNT(*) FROM `yourock_alias` INNER JOIN `yourock_achiever` ON (`yourock_alias`.`id` = `yourock_achiever`.`alias_id`) WHERE `yourock_achiever`.`achievement_id` = j0xvw9 '}

这就是我想要的,但=在我看来意味着 IS 是一个对象。如果对象在成就列表中任何地方,则django实际执行的查询将返回。

这是否正确设置并且是非常违反直觉还是我做错了什么?

2 个答案:

答案 0 :(得分:2)

我无法回答为什么设计决策以这种方式实现它,但很可能它遵循Python的理念,即事物应该明确指定而不是暗示。

它不起作用的原因是因为__contains需要引用一个字段。你传递的那个位是对整个对象的引用。

答案 1 :(得分:1)

在第二种情况下,您正在比较对象。并且,从Django documentation开始,为了比较您必须使用==运算符的对象。

最重要的是:为什么不使用ach.alias_set.objects.count(),如query section of the manual中所述?