Индивидуальный заказ в Джанго - PullRequest
30 голосов
/ 19 мая 2009

Как вы определяете конкретный порядок в Django QuerySet s?

В частности, если у меня есть QuerySet, например, так: ['a10', 'a1', 'a2'].

Обычный заказ (используя Whatever.objects.order_by('someField')) даст мне ['a1', 'a10', 'a2'], а я ищу: ['a1', 'a2', 'a10'].

Как правильно определить мою собственную технику заказа?

Ответы [ 4 ]

42 голосов
/ 19 мая 2009

Насколько мне известно, нет способа указать порядок на стороне базы данных таким образом, поскольку он будет слишком специфичным для бэкэнда. Вы можете прибегнуть к старой доброй сортировке Python:

class Foo(models.Model):
    name = models.CharField(max_length=128)

Foo.objects.create(name='a10')
Foo.objects.create(name='a1')
Foo.objects.create(name='a2')

ordered = sorted(Foo.objects.all(), key=lambda n: (n[0], int(n[1:])))
print ordered # yields a1, a2, 10

Если вам часто нужна такая сортировка, я бы порекомендовал создать собственный подкласс models.Manager для вашей модели, который выполняет упорядочение. Что-то вроде:

class FooManager(models.Manager):
    def in_a_number_order(self, *args, **kwargs):
        qs = self.get_query_set().filter(*args, **kwargs)
        return sorted(qs, key=lambda n: (n[0], int(n[1:])))

class Foo(models.Model):
    ... as before ...
    objects = FooManager()

print Foo.objects.in_a_number_order()
print Foo.objects.in_a_number_order(id__in=[5, 4, 3]) # or any filtering expression
29 голосов
/ 20 мая 2009

@ Ответ Джаррета (выполните сортировку в Python) отлично подходит для простых случаев. Как только у вас есть большая таблица, и вы хотите, скажем, извлечь только первую страницу результатов, отсортированных определенным образом, этот подход обрывается (вам нужно извлечь каждую строку из базы данных, прежде чем вы сможете выполнить сортировку). В этот момент я хотел бы добавить денормализованное поле «сортировка», которое вы заполняете из поля «имя» во время сохранения, которое можно отсортировать на уровне БД обычным способом.

0 голосов
/ 04 ноября 2014

Если у вас есть большие наборы данных и дополнительно используется бэкэнд SOLR (например, с Haystack):

Используйте solr.ICUCollationField с параметром numeric=true в качестве типа для полей сортировки. Это будет сортировать в соответствии с языком и, если числа присутствуют, будет сортировать числовую часть в соответствии с числовыми правилами вместо сортировки строк.

См: https://cwiki.apache.org/confluence/display/solr/Language+Analysis#LanguageAnalysis-UnicodeCollation http://www.solr -start.com / Javadoc / Solr-Lucene / орг / Apache / Solr / схема / ICUCollationField.html

0 голосов
/ 19 мая 2009

Это зависит от того, где вы хотите его использовать.

Если вы хотите использовать его в своих собственных шаблонах, я бы предложил написать шаблон-тег, который сделает заказ за вас В нем вы можете использовать любой алгоритм сортировки, который вы хотите использовать.

В админке я делаю пользовательскую сортировку, расширяя шаблоны под мои нужды и загружая тег шаблона, как описано выше

...