关于Q对象和外键的谜题

时间:2009-11-28 21:02:03

标签: python django django-models django-queryset django-q

我有一个这样的模型:

class Thing(models.Model):
    property1 = models.IntegerField()
    property2 = models.IntegerField()
    property3 = models.IntegerField()

class Subthing(models.Model):
    subproperty = models.IntegerField()
    thing = modelsForeignkey(Thing)
    main = models.BooleanField()

我有一个传递过滤器列表的函数,其中每个过滤器的格式为{'type':something,'value':x}。此函数需要返回一组结果并将所有过滤器组合在一起:

final_q = Q()
for filter in filters:
        q = None
        if filter['type'] =='thing-property1':
            q = Q(property1=filter['value'])
        elif filter['type'] =='thing-property2':
            q = Q(property2=filter['value'])
        elif filter['type'] =='thing-property2':
            q = Q(property3=filter['value'])
        if q:
            final_q = final_q & q
return Thing.objects.filter(final_q).distinct()

每个Subthing都有一个布尔属性'main'。每件事都有1个且只有1个Subthing,其中main == True。

我现在需要添加过滤器,该过滤器返回所有包含main==Truesubproperty==filter['value']

的子内容的内容

我可以将此作为我正在构建的Q对象的一部分吗?如果不是怎么回事?我在新过滤器之前得到的查询集可能非常大,所以我想要一个不涉及循环结果的方法。

2 个答案:

答案 0 :(得分:2)

如果你明确地给你的Subthings在他们与Thing的关系中给出了“related_name”,那就更容易理解了

class Subthing(models.Model):
    ...
    thing = models.ForeignKey(Thing, related_name='subthings')
    ...

现在,您使用Django join syntax来构建Q对象:

Q(subthings__main=True) & Q(subthings__subproperty=filter['value'])

反向关系有默认名称'subthing_set',但我发现如果你给它一个更好的名字,比如'subthings',它会更容易理解。

答案 1 :(得分:1)

使用(而不是开头的final_q=Q()

final_q=Q(subthing_set__main=True)
sub_vals = map(lambda v: v['value'], filters)
if sub_vals:
    final_q = final_q & Q(subthing_set__subproperty__in=sub_vals)

应该得到你想要的东西,你也可以调整你的循环来构建sub_vals列表并在循环之后应用它。

subthing_set是并自动添加相关字段添加到Thing以访问相关的Subthings。

您可以指定其他相关名称,例如

thing=models.ForeignKey(Thing,related_name='subthings')