django用q对象动态过滤

bab*_*eii 31 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. )
Run Code Online (Sandbox Code Playgroud)

我该如何创建此功能?

Ril*_*ins 67

您需要遍历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)
Run Code Online (Sandbox Code Playgroud)

这将为您提供与所有标签匹配的结果,如您的示例所示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)
Run Code Online (Sandbox Code Playgroud)

我测试了它,它似乎工作得很好.

编辑2:感谢Freenode上#django中的kezabelle最初的想法.


Max*_*Max 12

你可以这样使用:

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)
Run Code Online (Sandbox Code Playgroud)

通过这种方式,您可以使用动态生成的字段名称.你也可以使用Q.ANDAND状态.

  • @MichaelHoffmann我不太明白您的意思,也许是这样:Q(** {“ {} __ in” .format(key):value}) (2认同)