如何以与Django中订购QuerySets相同的方式订购列表?

时间:2009-04-15 20:57:35

标签: python django django-models

我的模型在其Meta类下有一个排序字段。当我执行查询并返回模型的QuerySet时,它按指定的顺序。但是,如果我在列表中有此模型的实例并在列表上执行排序方法,则顺序与我想要的顺序不同。有没有办法对模型的实例列表进行排序,使得顺序等于模型定义中指定的顺序?

3 个答案:

答案 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

如果您没有使用查询集,那么当然您仍然可以排序,就像任何人在python中对复杂对象进行排序一样:

  • 比较
  • 指定密钥
  • 装饰/排序/去除装饰

See the python wiki for a detailed explanation of all three.

答案 1 :(得分:3)

不是自动的,但有一点工作,是的。您需要在模型类上定义比较器函数(或 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)

答案 2 :(得分: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