如何以Django中QuerySets的方式排序列表?
我有一个模型,这个模型在它的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 中排序复杂对象是一样的:
- 比较器
- 指定键
- 装饰/排序/去装饰
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)