django用q对象动态过滤

时间:2012-10-25 20:34:29

标签: django django-models django-q

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

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

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

如何创建此功能?

5 个答案:

答案 0 :(得分:65)

您需要遍历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的最初想法。

答案 1 :(得分:8)

您可以这样使用:

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条件。

答案 2 :(得分:3)

首先准备一个标签列表,然后像这样查询:

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

答案 3 :(得分:1)

使用减少:

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]))

答案 4 :(得分:0)

您可能需要添加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)