如何以Django中QuerySets的方式排序列表?

2 投票
3 回答
3298 浏览
提问于 2025-04-15 11:08

我有一个模型,这个模型在它的Meta类里有一个排序字段。当我查询这个模型并得到一个查询集时,结果是按照我指定的顺序排列的。不过,如果我把这个模型的实例放在一个列表里,然后对这个列表使用排序方法,得到的顺序就和我想要的不同了。有没有办法让这个模型实例的列表按照模型定义的顺序来排序呢?

3 个回答

2

在Carl的回答基础上,你可以很简单地增加一个功能,让你可以使用所有的排序字段,甚至还能检测出哪些字段是反向排序的。

class Person(models.Model):
  first_name = models.CharField(max_length=50)
  last_name = models.CharField(max_length=50)
  birthday = date = models.DateField()

  class Meta:
    ordering = ['last_name', 'first_name']

  def __cmp__(self, other):
    for order in self._meta.ordering:
      if order.startswith('-'):
        order = order[1:]
        mode = -1
      else:
        mode = 1
      if hasattr(self, order) and hasattr(other, order):
        result = mode * cmp(getattr(self, order), getattr(other, order))
        if result: return result
    return 0
3

你的问题的答案是有点复杂,但大致上可以说是可以的,只是需要一些手动操作。如果你说的 list 是指通过一些复杂查询得到的 queryset,那当然可以:

queryset.order_by(ClassName.Meta.ordering)

或者

queryset.order_by(instance._meta.ordering)

或者

queryset.order_by("fieldname") #If you like being manual

如果你不是在处理 queryset,那当然也可以排序,方法和在 Python 中排序复杂对象是一样的:

  • 比较器
  • 指定键
  • 装饰/排序/去装饰

可以查看 Python 的维基,了解这三种方法的详细解释。

5

不是自动的,但只要稍微花点功夫,是可以做到的。你需要定义一个比较函数(或者在模型类中定义一个cmp方法),这个函数可以根据相关的属性来比较两个模型实例。比如:

class Dated(models.Model):
  ...
  created = models.DateTimeField(default=datetime.now)

  class Meta:
    ordering = ('created',)

  def __cmp__(self, other):
    try:
      return cmp(self.created, other.created)
    except AttributeError:
      return cmp(self.created, other)

撰写回答