Группировка списка кортежей Python - PullRequest
19 голосов
/ 12 февраля 2010

У меня есть список (label, count) кортежей, подобных этому:

[('grape', 100), ('grape', 3), ('apple', 15), ('apple', 10), ('apple', 4), ('banana', 3)]

Исходя из этого, я хочу суммировать все значения с одной и той же меткой (одинаковые метки всегда рядом) и вернуть список в том же порядке меток:

[('grape', 103), ('apple', 29), ('banana', 3)]

Я знаю, что мог бы решить что-то вроде:

def group(l):
    result = []
    if l:
        this_label = l[0][0]
        this_count = 0
        for label, count in l:
            if label != this_label:
                result.append((this_label, this_count))
                this_label = label
                this_count = 0
            this_count += count
        result.append((this_label, this_count))
    return result

Но есть ли более Pythonic / элегантный / эффективный способ сделать это?

Ответы [ 7 ]

30 голосов
/ 12 февраля 2010

itertools.groupby можете делать что хотите:

import itertools
import operator

L = [('grape', 100), ('grape', 3), ('apple', 15), ('apple', 10),
     ('apple', 4), ('banana', 3)]

def accumulate(l):
    it = itertools.groupby(l, operator.itemgetter(0))
    for key, subiter in it:
       yield key, sum(item[1] for item in subiter) 

>>> print list(accumulate(L))
[('grape', 103), ('apple', 29), ('banana', 3)]
>>> 
6 голосов
/ 12 февраля 2010

с помощью itertools и списочных представлений

import itertools

[(key, sum(num for _, num in value))
    for key, value in itertools.groupby(l, lambda x: x[0])]

Редактировать: , как указал Гнибблер: если l еще не отсортировано, замените его на sorted(l).

5 голосов
/ 12 февраля 2010
import collections
d=collections.defaultdict(int)
a=[]
alist=[('grape', 100), ('banana', 3), ('apple', 10), ('apple', 4), ('grape', 3), ('apple', 15)]
for fruit,number in alist:
    if not fruit in a: a.append(fruit)
    d[fruit]+=number
for f in a:
    print (f,d[f])

выход

$ ./python.py
('grape', 103)
('banana', 3)
('apple', 29)
4 голосов
/ 12 февраля 2010
>>> from itertools import groupby
>>> from operator import itemgetter
>>> L=[('grape', 100), ('grape', 3), ('apple', 15), ('apple', 10), ('apple', 4), ('banana', 3)]
>>> [(x,sum(map(itemgetter(1),y))) for x,y in groupby(L, itemgetter(0))]
[('grape', 103), ('apple', 29), ('banana', 3)]
2 голосов
/ 19 апреля 2017

моя версия без itertools
[(k, sum([y for (x,y) in l if x == k])) for k in dict(l).keys()]

0 голосов
/ 16 мая 2018

Метод

def group_by(my_list):
    result = {}
    for k, v in my_list:
        result[k] = v if k not in result else result[k] + v
    return result 

Использование

my_list = [
    ('grape', 100), ('grape', 3), ('apple', 15),
    ('apple', 10), ('apple', 4), ('banana', 3)
]

group_by(my_list) 

# Output: {'grape': 103, 'apple': 29, 'banana': 3}

Вы преобразуете в список кортежей, таких как list(group_by(my_list).items()).

0 голосов
/ 10 июля 2016

Или более простой и понятный ответ (без itertools):

pairs = [('foo',1),('bar',2),('foo',2),('bar',3)]

def sum_pairs(pairs):
  sums = {}
  for pair in pairs:
    sums.setdefault(pair[0], 0)
    sums[pair[0]] += pair[1]
  return sums.items()

print sum_pairs(pairs)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...