Django中的自定义排序

时间:2009-05-19 15:49:31

标签: python django django-models

如何在Django QuerySet中定义特定顺序?

具体来说,如果我有QuerySet这样的话:['a10', 'a1', 'a2']

定期订单(使用Whatever.objects.order_by('someField'))会给我['a1', 'a10', 'a2'],而我正在寻找:['a1', 'a2', 'a10']

定义我自己的排序技术的正确方法是什么?

4 个答案:

答案 0 :(得分:41)

据我所知,没有办法以这种方式指定数据库端排序,因为它过于特定于后端。您可能希望采用良好的老式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

答案 1 :(得分:28)

@Jarret的答案(在Python中排序)对于简单的情况非常有用。只要你有一个大表并且想要,例如,只拉动以某种方式排序的结果的第一页,这种方法就会中断(你必须从数据库中拉出每一行才能进行排序)。在那一点上,我将考虑添加一个非规范化的“排序”字段,您可以在保存时从“名称”字段填充该字段,您可以按照常规方式在数据库级别进行排序。

答案 2 :(得分:0)

这取决于您想要使用它的位置。

如果您想在自己的模板中使用它,我建议您编写一个模板标签,为您排序 在其中,您可以使用您想要使用的任何排序算法。

在管理员中,我通过扩展模板以满足我的需要并加载模板标签进行自定义排序

答案 3 :(得分:0)

如果您有更大的数据集并另外使用SOLR后端(例如使用Haystack):

solr.ICUCollationFieldnumeric=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