2017-03-08 31 views
1

难以忍受。设想一个数据框,其中包含多个产品类别“标签”,其中一些列表位于列表中,如下所示。PANDAS使用多个组在单列内使用多个组

import pandas as pd 

raw = { 
    'Products' : ['Rock On Leather Journal', 
        'Beats Earbuds In Ear Timer', 
        'Garmin 25mm Wristwatch' 
       ], 
    'Product Cost': [55,163,200], 
    'Product Category' : [['Music','Journals','Paper'], 
          ['Headphones','Music', 'Clocks'], 
          ['Watches','Clocks']] 
} 

data = pd.DataFrame(raw) 

什么是最好的方法来计算每个类别中有多少产品,并平均每个类别的成本?例如,

音乐:数量?,平均价格?

通常,这将是一个使用地图或groupby的简单过程。但类别列中列表的出现增加了一个棘手的转折。

回答

2

来源DF:

In [21]: data 
Out[21]: 
       Product Category Product Cost     Products 
0  [Music, Journals, Paper]   55  Rock On Leather Journal 
1 [Headphones, Music, Clocks]   163 Beats Earbuds In Ear Timer 
2   [Watches, Clocks]   200  Garmin 25mm Wristwatch 

首先让变换(平整)到以下DF:

In [22]: lst_col = 'Product Category' 
    ...: 
    ...: x = pd.DataFrame({ 
    ...:  col:np.repeat(data[col].values, data[lst_col].str.len()) 
    ...:  for col in data.columns.difference([lst_col]) 
    ...: }).assign(**{lst_col:np.concatenate(data[lst_col].values)})[data.columns.tolist()] 
    ...: 

In [23]: x 
Out[23]: 
    Product Category Product Cost     Products 
0   Music   55  Rock On Leather Journal 
1   Journals   55  Rock On Leather Journal 
2   Paper   55  Rock On Leather Journal 
3  Headphones   163 Beats Earbuds In Ear Timer 
4   Music   163 Beats Earbuds In Ear Timer 
5   Clocks   163 Beats Earbuds In Ear Timer 
6   Watches   200  Garmin 25mm Wristwatch 
7   Clocks   200  Garmin 25mm Wristwatch 

现在我们可以很容易地"count of how many Products within each Category, and to average the costs for each category"

In [25]: x.groupby('Product Category')['Product Cost'].agg(['size', 'mean']).reset_index() 
Out[25]: 
    Product Category size mean 
0   Clocks  2 181.5 
1  Headphones  1 163.0 
2   Journals  1 55.0 
3   Music  2 109.0 
4   Paper  1 55.0 
5   Watches  1 200.0 

几点说明:

利用这些信息,我们可以复制所有非列表列如下

In [7]: data[lst_col].str.len() 
Out[7]: 
0 3 
1 3 
2 2 
Name: Product Category, dtype: int64 

:每排

数列表中的元素现在

In [3]: x = pd.DataFrame({ 
    ...:  col:np.repeat(data[col].values, data[lst_col].str.len()) 
    ...:  for col in data.columns.difference([lst_col]) 
    ...: }) 

In [4]: x 
Out[4]: 
    Product Cost     Products 
0   55  Rock On Leather Journal 
1   55  Rock On Leather Journal 
2   55  Rock On Leather Journal 
3   163 Beats Earbuds In Ear Timer 
4   163 Beats Earbuds In Ear Timer 
5   163 Beats Earbuds In Ear Timer 
6   200  Garmin 25mm Wristwatch 
7   200  Garmin 25mm Wristwatch 

我们可以添加扁平list column

In [8]: np.concatenate(data[lst_col].values) 
Out[8]: 
array(['Music', 'Journals', 'Paper', 'Headphones', 'Music', 'Clocks', 'Watches', 'Clocks'], 
     dtype='<U10') 

In [5]: x.assign(**{lst_col:np.concatenate(data[lst_col].values)}) 
Out[5]: 
    Product Cost     Products Product Category 
0   55  Rock On Leather Journal   Music 
1   55  Rock On Leather Journal   Journals 
2   55  Rock On Leather Journal   Paper 
3   163 Beats Earbuds In Ear Timer  Headphones 
4   163 Beats Earbuds In Ear Timer   Music 
5   163 Beats Earbuds In Ear Timer   Clocks 
6   200  Garmin 25mm Wristwatch   Watches 
7   200  Garmin 25mm Wristwatch   Clocks 

最后我们简单地选择原始顺序中的列:

In [6]: x.assign(**{lst_col:np.concatenate(data[lst_col].values)})[data.columns.tolist()] 
Out[6]: 
    Product Category Product Cost     Products 
0   Music   55  Rock On Leather Journal 
1   Journals   55  Rock On Leather Journal 
2   Paper   55  Rock On Leather Journal 
3  Headphones   163 Beats Earbuds In Ear Timer 
4   Music   163 Beats Earbuds In Ear Timer 
5   Clocks   163 Beats Earbuds In Ear Timer 
6   Watches   200  Garmin 25mm Wristwatch 
7   Clocks   200  Garmin 25mm Wristwatch 
+0

这个工程!尽管如此,我仍然试图通过.assign()完全理解“for”col部分发生的情况。它看起来像是每个类别发生的事情,您正在将行数据复制到一个新行中,以便每行都有一个类别。然后,使用.assign()添加所有其他列。但也许我错了。这比我见过的任何事情都要复杂得多(尽管很棒),我希望你会为看到这篇文章的其他人解释一下。 – Adestin

+0

@Adestin,我已经添加了一些解释 - 请检查 – MaxU

0

这取决于你的情况。如果它是这个大小,你可能想为每个元素创建一个布尔列 - 例如

unique_products = set(chain(*data['Product Category'])) 
for product in unique_products: 
    data['product_{}.format(product) = data.Products.apply(lambda x: product in x) 

或者,如果您有大量产品,请根据需要继续使用data.Product.apply(lamba x: product in x)。您也可以使用data.Product.isin([product_one, product_two])执行类似的检查。

一旦你有一个合成列,你可以使用它合并。