web-dev-qa-db-ja.com

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

このフィーチャーを作成するにはどうすればよいですか?

32
babbaggeii

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で示されているように、allタグに一致する結果を提供します。実際に代わりに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の最初のアイデアのクレジット。

66
Riley Watkins

次のように使用できます。

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

11
Max

最初にタグリストを準備し、次のようにクエリします。

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