关于 Q 对象和外键的谜题
Posted
技术标签:
【中文标题】关于 Q 对象和外键的谜题【英文标题】:A puzzle concerning Q objects and Foreign Keys 【发布时间】:2010-12-21 06:57:51 【问题描述】:我有一个这样的模型:
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 个子事物,其中 main==True。
我现在需要添加过滤器,返回所有具有 main==True
和 subproperty==filter['value']
的子事物的事物
我可以将其作为我正在构建的Q
对象的一部分吗?如果没有怎么办?在我的新过滤器之前获得的查询集可能非常大,所以我想要一种不涉及循环结果的方法。
【问题讨论】:
有错别字,应该是models.ForeignKey() 【参考方案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 以访问相关的 Subthing。
您可以指定另一个相关名称,例如
thing=models.ForeignKey(Thing,related_name='subthings')
【讨论】:
【参考方案2】:如果您在子事物与事物的关系中明确地为您的子事物指定一个“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”,它会更容易理解。
【讨论】:
噢!出于某种原因,我一直在寻找比这更复杂的解决方案!这似乎工作得很好。以上是关于关于 Q 对象和外键的谜题的主要内容,如果未能解决你的问题,请参考以下文章