Существует множество способов написания программы на Python, которая вычисляет гистограмму.
Под гистограммой я подразумеваю функцию, которая считает количество объектов в iterable
и выводит значения в словаре. Например:
>>> L = 'abracadabra'
>>> histogram(L)
{'a': 5, 'b': 2, 'c': 1, 'd': 1, 'r': 2}
Один из способов написать эту функцию:
def histogram(L):
d = {}
for x in L:
if x in d:
d[x] += 1
else:
d[x] = 1
return d
Существуют ли более краткие способы написания этой функции?
Если бы у нас были словарные выражения в Python, мы могли бы написать:
>>> { x: L.count(x) for x in set(L) }
но поскольку в Python 2.6 их нет, мы должны написать:
>>> dict([(x, L.count(x)) for x in set(L)])
Хотя этот подход может быть читабельным, он неэффективен: L обходится несколько раз. Кроме того, это не будет работать для генераторов с одиночной жизнью; функция должна одинаково хорошо работать для генераторов итераторов, таких как:
def gen(L):
for x in L:
yield x
Мы можем попытаться использовать функцию reduce
(R.I.P.):
>>> reduce(lambda d,x: dict(d, x=d.get(x,0)+1), L, {}) # wrong!
Упс, это не работает: имя ключа 'x'
, а не x
. (
Я закончил с:
>>> reduce(lambda d,x: dict(d.items() + [(x, d.get(x, 0)+1)]), L, {})
(В Python 3 нам нужно было бы написать list(d.items())
вместо d.items()
, но это гипотетически, поскольку там нет reduce
.)
Пожалуйста, побейте меня лучшей, более читаемой строчкой! ;)