我正在尝试根据用户输入标签查询数据库。标签的数量可以是 0-5,所以我需要动态创建查询。
所以我有一个标签列表,tag_list,我想查询数据库:
design_list = Design.objects.filter(Q(tags__tag__contains = "tag1") and Q(tags__tag__contains = "tag2") and etc. etc. )
如何创建此功能?
我正在尝试根据用户输入标签查询数据库。标签的数量可以是 0-5,所以我需要动态创建查询。
所以我有一个标签列表,tag_list,我想查询数据库:
design_list = Design.objects.filter(Q(tags__tag__contains = "tag1") and Q(tags__tag__contains = "tag2") and etc. etc. )
如何创建此功能?
您需要遍历 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 最初的想法。
你可以这样使用:
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
条件。
只需先准备一个标签列表,查询如下:
tags = ['tag1', 'tag2',...]
design_list = Design.objects.filter(tags__tag__contains__in = tags)
您可能需要添加 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)
使用减少:
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]))