Python:如何按几个值对字典列表进行排序?

Jok*_*oko 61 python sorting list

我想首先按值排序,然后按第二个值排序.是否有捷径可寻?这是一个小例子:

A = [{'name':'john','age':45},
     {'name':'andi','age':23},
     {'name':'john','age':22},
     {'name':'paul','age':35},
     {'name':'john','age':21}]
Run Code Online (Sandbox Code Playgroud)

此命令用于对此列表进行排序'name':

sorted(A, key = lambda user: user['name'])
Run Code Online (Sandbox Code Playgroud)

但我如何按第二个值对此列表进行排序?就像'age'在这个例子中.

我想要这样的排序(首先排序'name'然后排序'age'):

andi - 23
john - 21
john - 22
john - 45
paul - 35
Run Code Online (Sandbox Code Playgroud)

谢谢!

jam*_*lak 89

>>> A = [{'name':'john','age':45},
     {'name':'andi','age':23},
     {'name':'john','age':22},
     {'name':'paul','age':35},
     {'name':'john','age':21}]
>>> sorted(A, key = lambda user: (user['name'], user['age']))
[{'age': 23, 'name': 'andi'}, {'age': 21, 'name': 'john'}, {'age': 22, 'name': 'john'}, {'age': 45, 'name': 'john'}, {'age': 35, 'name': 'paul'}]
Run Code Online (Sandbox Code Playgroud)

这由两个属性的元组排序,以下是等效的,更快/更清洁:

>>> from operator import itemgetter
>>> sorted(A, key=itemgetter('name', 'age'))
[{'age': 23, 'name': 'andi'}, {'age': 21, 'name': 'john'}, {'age': 22, 'name': 'john'}, {'age': 45, 'name': 'john'}, {'age': 35, 'name': 'paul'}]
Run Code Online (Sandbox Code Playgroud)

来自评论:@Bakuriu

我敢打赌,两者之间没有太大的区别,但是itemgetter避免了一些开销,因为它提取了密钥并tuple在单个操作码(CALL_FUNCTION)期间进行,而调用lambda将需要调用函数,加载各种常量(这是其他字节码)最后调用下标(BINARY_SUBSCR),构建tuple并返回它......这对解释器来说还有很多工作要做.

总结一下:itemgetter将执行完全保持在C关卡上,因此它尽可能快.

  • @catchmeifyoutry我敢打赌,两者之间没有*大*差异,但是`itemgetter'避免了一些开销,因为它在单个操作码(`CALL_FUNCTION`)中提取键并生成元组,而调用lambda将有调用函数,加载各种常量(这是其他字节码)最后调用下标(`BINARY_SUBSCR`),构建元组并返回它...这对解释器来说还有很多工作要做 (4认同)
  • 我会对解释*为什么*itemgetter比lambda表达式快得多感兴趣.它不归结为相同的查找? (3认同)
  • @catchmeifyoutry是的.请注意,文档说它与*python代码*等效*,而不是以这种方式实现.特别是`itemgetter`实际上是一个`type`(参见`type(operator.itemgetter)`)所以它甚至不是一个真正的函数.这是典型的:许多内置函数在C中实现,通常它们是类型而不是函数(例如`enumerate`). (3认同)

Jon*_*nts 53

from operator import itemgetter

sorted(your_list, key=itemgetter('name', 'age'))
Run Code Online (Sandbox Code Playgroud)

  • 你是否想知道`operator.itemgetter`是否能做出几乎神奇的事情?它不是.它返回一个元组(在这种情况下长度为2),然后`sorted`执行*正确的事情*. (2认同)