Django получает QuerySet из массива идентификаторов в определенном порядке - PullRequest
60 голосов
/ 07 февраля 2011

Вот быстрый для вас:

У меня есть список идентификаторов, которые я хочу использовать для возврата QuerySet (или массива, если необходимо), но я хочу сохранить этот порядок.

Спасибо

Ответы [ 5 ]

133 голосов
/ 06 июня 2016

Начиная с Django 1.8, вы можете делать:

from django.db.models import Case, When

pk_list = [10, 2, 1]
preserved = Case(*[When(pk=pk, then=pos) for pos, pk in enumerate(pk_list)])
queryset = MyModel.objects.filter(pk__in=pk_list).order_by(preserved)
56 голосов
/ 07 февраля 2011

Я не думаю, что вы можете применить этот конкретный порядок на уровне базы данных, поэтому вам нужно сделать это в python.

id_list = [1, 5, 7]
objects = Foo.objects.filter(id__in=id_list)

objects = dict([(obj.id, obj) for obj in objects])
sorted_objects = [objects[id] for id in id_list]

При этом создается словарь объектов с их идентификатором в качестве ключа, поэтому их можно легко найти при создании отсортированного списка.

26 голосов
/ 12 декабря 2012

Если вы хотите сделать это с помощью in_bulk, вам действительно нужно объединить два ответа выше:

id_list = [1, 5, 7]
objects = Foo.objects.in_bulk(id_list)
sorted_objects = [objects[id] for id in id_list]

В противном случае результатом будет словарь, а не специально упорядоченный список.

21 голосов
/ 25 августа 2014

Вот способ сделать это на уровне базы данных. Скопируйте пасту из: blog.mathieu-leplatre.info :

MySQL

SELECT *
FROM theme
ORDER BY FIELD(`id`, 10, 2, 1);

То же самое с Джанго:

pk_list = [10, 2, 1]
ordering = 'FIELD(`id`, %s)' % ','.join(str(id) for id in pk_list)
queryset = Theme.objects.filter(pk__in=[pk_list]).extra(
           select={'ordering': ordering}, order_by=('ordering',))

PostgreSQL :

SELECT *
FROM theme
ORDER BY
  CASE
    WHEN id=10 THEN 0
    WHEN id=2 THEN 1
    WHEN id=1 THEN 2
  END;

То же самое с Джанго:

pk_list = [10, 2, 1]
clauses = ' '.join(['WHEN id=%s THEN %s' % (pk, i) for i, pk in enumerate(pk_list)])
ordering = 'CASE %s END' % clauses
queryset = Theme.objects.filter(pk__in=pk_list).extra(
           select={'ordering': ordering}, order_by=('ordering',))
9 голосов
/ 30 ноября 2015
id_list = [1, 5, 7]
objects = Foo.objects.filter(id__in=id_list)
sorted(objects, key=lambda i: id_list.index(i.pk))
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...