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 ответов

Решение

Вы захотите перебрать 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: Кредит кезабель в #django на Freenode для первоначальной идеи.

Вы можете использовать этот способ:

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)

Возможно, вам потребуется добавить условия И и ИЛИ

    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]))
Другие вопросы по тегам