Python operator.itemgetter()
一、问题
对列表、元组、字典或(字典、元组组成的列表),根据属性值排序。
二、解决方案
1. 列表、元组、字典
operator.itemgetter()
获取的不是值,而是定义一个函数,通过该函数作用到对象上才能获取值。
from operator import itemgetter
a = [1, 2, 3]
b = itemgetter(1) # 定义函数b,获取对象的第1个值
c = itemgetter(1, 0) # 定义函数c,获取对象的第1、0个值
print(b(a))
print(c(a))
a = {'name': 'wangke', 'age':18, 'sex': 'man'}
b = itemgetter('age')
c = itemgetter('name', 'age')
print(b(a))
print(c(a))
输出:
2
(2, 1)
18
('wangke', 18)
2. 字典列表
operator.itemgetter()
可以排序这样的数据结构。
数据:
rows = [
{'fname': 'Brian', 'lname': 'Jones', 'uid': 1003},
{'fname': 'David', 'lname': 'Beazley', 'uid': 1002},
{'fname': 'John', 'lname': 'Cleese', 'uid': 1001},
{'fname': 'Big', 'lname': 'Jones', 'uid': 1004}
]
根据字典字段排序:
from operator import itemgetter
rows_fname = sorted(rows, key=itemgetter('fname'))
rows_uid = sorted(rows, key=itemgetter('uid'))
print(rows_fname)
print(rows_uid)
输出:
[{'fname': 'Big', 'lname': 'Jones', 'uid': 1004},
{'fname': 'Brian', 'lname': 'Jones', 'uid': 1003},
{'fname': 'David', 'lname': 'Beazley', 'uid': 1002},
{'fname': 'John', 'lname': 'Cleese', 'uid': 1001}]
[{'fname': 'John', 'lname': 'Cleese', 'uid': 1001},
{'fname': 'David', 'lname': 'Beazley', 'uid': 1002},
{'fname': 'Brian', 'lname': 'Jones', 'uid': 1003},
{'fname': 'Big', 'lname': 'Jones', 'uid': 1004}]
itemgetter()
也可以用lambda
代替,但是,itemgetter()
会稍微快点。。
rows_fname = sorted(rows, key=lambda r: r['fanme'])
rows_uid = sorted(rows, key=lambda r: (r['fanme'], r['uid']))
3. 元组列表
from operator import itemgetter
a = [('john', 'A', 15), ('jane', 'B', 12), ('dave', 'C', 10)]
b = sorted(a, key=itemgetter(1))
c = sorted(a, key=itemgetter(2))
print(b)
print(c)
输出:
[('john', 'A', 15), ('jane', 'B', 12), ('dave', 'C', 10)]
[('dave', 'C', 10), ('jane', 'B', 12), ('john', 'A', 15)]
三、讨论
查找的元素个数相对比较小,用 nsmallest()、nlargest()。
查找元素个数与总的元素个数接近,通常先排序,然后再切片。sorted()。