Django中的自定义排序

Yuv*_*dam 30 python django django-models

你如何在Django中定义一个特定的顺序QuerySet

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

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

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

Jar*_*die 42

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

如果您发现自己需要这种排序,我建议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
Run Code Online (Sandbox Code Playgroud)

  • 我担心`sorted`不会返回查询集; 它是一个内置的python函数,所以它对Django一无所知.Django的查询集为迭代实现了python的接口,因此`sorted`能够对它们起作用,因为它认为它刚刚传递了一个列表.如果你需要返回一个QS,AFAIK没有简单的方法来创建一个排序返回的列表中的QS ......它可以伪造一个QS(基本上实现它的接口),但这可能很棘手.有很多Manager方法没有返回QS,我担心这个排序功能会与那些一致. (4认同)
  • 尝试此方法时出现此错误::'Model'对象是不可索引的 (2认同)

Car*_*yer 31

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

  • +1用于解决与绩效相关的问题; 事实上,你的建议是我们用于报告的一些数据,在数据库不可知的环境中,我们已经有数千万行,所以我可以保证它运行良好. (3认同)