Пользовательский заказ в Django
Как вы определяете конкретный порядок в Django QuerySet
s?
В частности, если у меня есть QuerySet
так: ['a10', 'a1', 'a2']
.
Обычный порядок (используя Whatever.objects.order_by('someField')
) даст мне ['a1', 'a10', 'a2']
, в то время как я ищу: ['a1', 'a2', 'a10']
.
Каков правильный способ определить мою собственную методику заказа?
Ответы
Ответ 1
Насколько мне известно, нет способа указать заказ на стороне базы данных таким образом, поскольку он будет слишком специфичным для бэкэнд. Вы можете прибегнуть к хорошей старомодной сортировке 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
Ответ 2
@Jarret ответ (сделайте вид в Python) отлично работает для простых случаев. Как только у вас будет большой стол и вы хотите, скажем, вытащить только первую страницу результатов, отсортированную определенным образом, этот подход ломается (вам нужно вытащить каждую строку из базы данных, прежде чем вы сможете сделать сортировку). В этот момент я рассмотрю добавление денормализованного поля "сортировки", которое вы заполняете из поля "имя" во время сохранения, которое можно обычным образом сортировать на уровне DB.
Ответ 3
Это зависит от того, где вы хотите его использовать.
Если вы хотите использовать его в своих собственных шаблонах, я бы предложил написать шаблон-тег, который сделает заказ для вас
В нем вы можете использовать любой алгоритм сортировки, который вы хотите использовать.
В admin я делаю пользовательскую сортировку, расширяя шаблоны до моих потребностей и загружая шаблон-тег, как описано выше
Ответ 4
Если у вас есть большие наборы данных и дополнительно используйте бэкэнд 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/org/apache/solr/schema/ICUCollationField.html