我有一个如下列表。
[['Andrew', '1', '9'], ['Peter', '1', '10'], ['Andrew', '1', '8'], ['Peter', '1', '11'], ['Sam', '4', '9'], ['Andrew', '2', '2']]
我想总结按其他列分组的最后一列。结果就像这样
[['Andrew', '1', '17'], ['Peter', '1', '21'], ['Sam', '4', '9'], ['Andrew', '2', '2']]
仍然是一个清单。
在实际操作中,我总是想总结由许多其他列分组的最后一列。有没有办法在Python中做到这一点?非常感谢。
答案 0 :(得分:14)
除了最后一列之外的所有列动态分组:
In [24]: df = pd.DataFrame(data)
In [25]: df.groupby(df.columns[:-1].tolist(), as_index=False).agg(lambda x: x.astype(int).sum()).values.tolist()
Out[25]: [['Andrew', '1', 17], ['Andrew', '2', 2], ['Peter', '1', 21], ['Sam', '4', 9]]
答案 1 :(得分:10)
这是通过collections.defaultdict
的O(n)解决方案,适用于任意数量的密钥。
如果您想要的输出是一个列表,那么这可能比通过Pandas的解决方案更可取,这需要转换为非标准类型。
from collections import defaultdict
lst = [['Andrew', '1', '9'], ['Peter', '1', '10'], ['Andrew', '1', '8'],
['Peter', '1', '11'], ['Sam', '4', '9'], ['Andrew', '2', '2']]
d = defaultdict(int)
for *keys, val in lst:
d[tuple(keys)] += int(val)
res = [[*k, v] for k, v in sorted(d.items())]
<强>结果强>
[['Andrew', '1', 17], ['Andrew', '2', 2], ['Peter', '1', 21], ['Sam', '4', 9]]
<强>解释强>
defaultdict
列表中。答案 2 :(得分:9)
OP1
您可以传递索引sum
并将tolist转换回列表
pd.DataFrame(L).\
set_index([0,1])[2].astype(int).sum(level=[0,1]).\
reset_index().values.tolist()
Out[78]: [['Andrew', '1', 17], ['Peter', '1', 21], ['Sam', '4', 9], ['Andrew', '2', 2]]
Op2中
有关列表的列表,您可以使用groupby
itertools
from itertools import groupby
[k+[sum(int(v) for _,_, v in g)] for k, g in groupby(sorted(l), key = lambda x: [x[0],x[1]])]
Out[98]: [['Andrew', '1', 17], ['Andrew', '2', 2], ['Peter', '1', 21], ['Sam', '4', 9]]
答案 3 :(得分:7)
创建到DataFrame
并按第一列和第二列聚合第三列转换为整数,最后转换回list
s:
df = pd.DataFrame(L)
L = df[2].astype(int).groupby([df[0], df[1]]).sum().reset_index().values.tolist()
print (L)
[['Andrew', '1', 17], ['Andrew', '2', 2], ['Peter', '1', 21], ['Sam', '4', 9]]
只有defaultdict和python 3.x的解决方案:
from collections import defaultdict
d = defaultdict(int)
#https://stackoverflow.com/a/10532492
for *head, tail in L:
d[tuple(head)] += int(tail)
d = [[*i, j] for i, j in sorted(d.items())]
print (d)
[['Andrew', '1', 17], ['Andrew', '2', 2], ['Peter', '1', 21], ['Sam', '4', 9]]
答案 4 :(得分:6)
pd.factorize
和np.bincount
f, u = pd.factorize([tuple(x[:-1]) for x in data])
v = np.array([x[-1] for x in data], int)
[list(k) + [int(v)] for k, v in zip(u, np.bincount(f, v))]
[['Andrew', '1', 17], ['Peter', '1', 21], ['Sam', '4', 9], ['Andrew', '2', 2]]
答案 5 :(得分:0)
加了我的2美分。这两个都使用groupby
,agg
V1:引入了新的求和函数。
sum=lambda x: x.astype(int).sum()
print(df.groupby([0,1], as_index=False).agg({2: sum}).values.tolist())
V2:将列转换为数字
df[2] = pd.to_numeric(df[2])
print(df.groupby([0,1], as_index=False).agg({2: sum}).values.tolist())
将返回:
[['Andrew', '1', 17], ['Andrew', '2', 2], ['Peter', '1', 21], ['Sam', '4', 9]]