我有一个像这样的数据框df1
id ` text c1
1 Hello world how are you people 1
2 Hello people I am fine people 1
3 Good Morning people -1
4 Good Evening -1
我要使df2
如此,它只包含df1
的所有单词一次,其计数(出现的总数)和
我想对c1
列求和并在df2
中加一个新列(仅当该行中有一个单词时才求和)。
预期输出:
Word Totalcount Points
hello 2 2
world 1 1
how 1 1
are 1 1
you 1 1
people 3 1
I 1 1
am 1 1
fine 1 1
Good 2 -2
Morning 1 -1
Evening 1 -1
答案 0 :(得分:7)
首先通过DataFrame.pop
,Series.str.split
,DataFrame.stack
将Series
和DataFrame.join
的列提取为原始列,然后通过DataFrame.drop_duplicates
删除重复项并进行汇总由GroupBy.agg
加上计数和sum
:
s = (df.pop('text')
.str.split(expand=True)
.stack()
.reset_index(1, drop=True)
.rename('text'))
df1 = (df.join(s)
.reset_index(drop=True)
.drop_duplicates(['id','text'])
.groupby('text', sort=False)['c1']
.agg([('Totalcount','size'),('Points','sum')])
.reset_index()
.rename(columns={'text':'Word'}))
print (df1)
Word Totalcount Points
0 Hello 2 2
1 world 1 1
2 how 1 1
3 are 1 1
4 you 1 1
5 people 3 1
6 I 1 1
7 am 1 1
8 fine 1 1
9 Good 2 -2
10 Morning 1 -1
11 Evening 1 -1
编辑:
为获得更好的性能,请结合使用chain.from_iterable
和numpy.repeat
:
from itertools import chain
splitted = [x.split() for x in df['text']]
lens = [len(x) for x in splitted]
df = pd.DataFrame({
'Word' : list(chain.from_iterable(splitted)),
'id' : df['id'].values.repeat(lens),
'c1' : df['c1'].values.repeat(lens)
})
df1 = (df.drop_duplicates(['id','Word'])
.groupby('Word', sort=False)['c1']
.agg([('Totalcount','size'),('Points','sum')])
.reset_index())
print (df1)
Word Totalcount Points
0 Hello 2 2
1 world 1 1
2 how 1 1
3 are 1 1
4 you 1 1
5 people 3 1
6 I 1 1
7 am 1 1
8 fine 1 1
9 Good 2 -2
10 Morning 1 -1
11 Evening 1 -1