Django 动态过滤使用 Q 对象。

55

我正在尝试基于用户输入的标签查询数据库。标签数量可以是0-5个,因此我需要动态创建查询。

所以我有一个标签列表tag_list,我想查询数据库:

design_list = Design.objects.filter(Q(tags__tag__contains = "tag1") and Q(tags__tag__contains = "tag2") and etc. etc. )

我该如何创建这个功能?

5个回答

125
您需要循环遍历tag_list并为每个标签应用过滤器。
tag_list = ['tag1', 'tag2', 'tag3']
base_qs = Design.objects.all()
for t in tag_list:
    base_qs = base_qs.filter(tags__tag__contains=t)

这将给你与示例中的and相匹配的所有标签结果。如果实际上你需要or,那么你可能需要使用Q对象。 编辑:我现在认为我已经找到了你要找的内容。
tags = ['tag1', 'tag2', 'tag3']
q_objects = Q() # Create an empty Q object to start with
for t in tags:
    q_objects |= Q(tags__tag__contains=t) # 'or' the Q objects together

designs = Design.objects.filter(q_objects)

我测试了一下,似乎效果非常好。 编辑2:感谢Freenode上Django频道的kezabelle提供的初始想法。

谢谢,那很有道理。我已经尝试使用Q对象,但它没有返回正确的项目。这里有什么我做错了吗:design_list = Design.objects.all() for t in tag_list: design_list = design_list.filter(Q(tags__tag__contains = t))。当只有一个标签时它可以工作,但是多个标签就不行了。 - babbaggeii
在你的代码中,Q对象没有起到任何作用。你只是创建了一个查询集,最终看起来像 Design.objects.filter(tags__tag__contains='tag1').filter(tags__tag__contains='tag2') 等等。你可能想要的是 Design.objects.filter(Q(tags__tag__contains='tag1') | Q(tags__tag__contains='tag2')),但以一种方式给你一个可变数量的Q对象。 - Riley Watkins
好的,那就是我需要寻找的。 - babbaggeii
如果您希望在tags为空时,designs也为空,则可以使用Q(pk__in=[])作为q_objects的起始值。 - Jonathan Richards
很遗憾,QuerySet API没有一个“any”方法,它可以采用具有“或”语义的Q对象列表,就像filter/exclude/get可以采用具有“和”语义的Q对象列表一样。 - odigity
我是新手,但我成功地让两个示例运行起来了:https://gist.github.com/odigity/53bf47cca9a238a905c276760153838d - odigity

40

你可以使用这种方式:

my_dict = {'field_1': 1, 'field_2': 2, 'field_3': 3, ...}  # Your dict with fields
or_condition = Q()
for key, value in my_dict.items():
    or_condition.add(Q(**{key: value}), Q.OR)

query_set = MyModel.objects.filter(or_condition)

通过这种方式,您可以使用动态生成的字段名称。同时,您也可以使用Q.AND来表示AND条件。


有没有办法使用__in来实现这个?比如获取一个匹配字段和值列表的Q,其中字段名和值列表由变量给出而不是字面值? - Michael Hoffmann
4
Q(**{"{}__in".format(key): value}) 的意思可能是:根据键值对的方式,查询具有特定值的对象。 - Max

3

首先,准备一个标签列表,然后进行查询:

tags = ['tag1', 'tag2',...]
design_list = Design.objects.filter(tags__tag__contains__in = tags)

2
你可能需要添加AND和OR条件。
    query = (Q(fild1='ENABLE'))
    # Filter by list
    query.add(Q(fild2__in=[p.code for p in Objects.field.all()]), Q.AND)

    # filter OR
    q_objects = Q(field3='9999999')
    for x in myList:
        q_objects.add(Q(field3=x.code), Q.OR)

    query.add(q_objects, Q.AND)

1
使用 reduce:
from functools import reduce
design_list = Design.objects.filter(reduce(lambda q1,q2: q1 & q2,
                                           [Q(tags__tag__contains=t)
                                            for t in tag_list]))

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接