pandas convertem algumas colunas em linhas

115

Portanto, meu conjunto de dados tem algumas informações por localização para n datas. O problema é que cada data é, na verdade, um cabeçalho de coluna diferente. Por exemplo, o CSV parece

location    name    Jan-2010    Feb-2010    March-2010
A           "test"  12          20          30
B           "foo"   18          20          25

O que eu gostaria é que se parecesse

location    name    Date        Value
A           "test"  Jan-2010    12       
A           "test"  Feb-2010    20
A           "test"  March-2010  30
B           "foo"   Jan-2010    18       
B           "foo"   Feb-2010    20
B           "foo"   March-2010  25

O problema é que não sei quantas datas estão na coluna (embora eu saiba que sempre começarão após o nome)

Wizuriel
fonte

Respostas:

207

ATUALIZAÇÃO
De v0.20, melté uma função de primeira ordem, agora você pode usar

df.melt(id_vars=["location", "name"], 
        var_name="Date", 
        value_name="Value")

  location    name        Date  Value
0        A  "test"    Jan-2010     12
1        B   "foo"    Jan-2010     18
2        A  "test"    Feb-2010     20
3        B   "foo"    Feb-2010     20
4        A  "test"  March-2010     30
5        B   "foo"  March-2010     25

VERSÕES ANTIGAS (ER): <0,20

Você pode usar pd.meltpara obter a maior parte do caminho até lá e depois classificar:

>>> df
  location  name  Jan-2010  Feb-2010  March-2010
0        A  test        12        20          30
1        B   foo        18        20          25
>>> df2 = pd.melt(df, id_vars=["location", "name"], 
                  var_name="Date", value_name="Value")
>>> df2
  location  name        Date  Value
0        A  test    Jan-2010     12
1        B   foo    Jan-2010     18
2        A  test    Feb-2010     20
3        B   foo    Feb-2010     20
4        A  test  March-2010     30
5        B   foo  March-2010     25
>>> df2 = df2.sort(["location", "name"])
>>> df2
  location  name        Date  Value
0        A  test    Jan-2010     12
2        A  test    Feb-2010     20
4        A  test  March-2010     30
1        B   foo    Jan-2010     18
3        B   foo    Feb-2010     20
5        B   foo  March-2010     25

(Pode ser necessário adicionar um .reset_index(drop=True), apenas para manter a saída limpa.)

Nota : pd.DataFrame.sort foi descontinuado a favor de pd.DataFrame.sort_values.

DSM
fonte
@DSM qual seria o inverso desta função. ou seja, como alguém seria convertido df2[de volta] paradf
3kstc
1
@ 3kstc Tente aqui ou aqui . Você está querendo olhar para os pivôs. Possivelmente pandas.pivot_table(df2,values='Value',index=['location','name'],columns='Date').reset_index().
Teepeemm
1
@DSM existe alguma maneira de voltar atrás? O que significa que tenho muitas linhas com o mesmo nome e gostaria que todas as datas estivessem em colunas diferentes
Adrian
17

Use set_indexcom stackpara MultiIndex Seriese para DataFrameadicionar reset_indexcom rename:

df1 = (df.set_index(["location", "name"])
         .stack()
         .reset_index(name='Value')
         .rename(columns={'level_2':'Date'}))
print (df1)
  location  name        Date  Value
0        A  test    Jan-2010     12
1        A  test    Feb-2010     20
2        A  test  March-2010     30
3        B   foo    Jan-2010     18
4        B   foo    Feb-2010     20
5        B   foo  March-2010     25
jezrael
fonte
5

Acho que encontrei uma solução mais simples

temp1 = pd.melt(df1, id_vars=["location"], var_name='Date', value_name='Value')
temp2 = pd.melt(df1, id_vars=["name"], var_name='Date', value_name='Value')

Concate todo temp1com temp2a coluna dename

temp1['new_column'] = temp2['name']

Agora você tem o que pediu.

Prometeu
fonte
4

pd.wide_to_long

Você pode adicionar um prefixo às colunas do ano e, em seguida, alimentar diretamente para pd.wide_to_long. Não vou fingir que isso é eficiente , mas em certas situações pode ser mais conveniente do que pd.melt, por exemplo, quando suas colunas já têm um prefixo apropriado.

df.columns = np.hstack((df.columns[:2], df.columns[2:].map(lambda x: f'Value{x}')))

res = pd.wide_to_long(df, stubnames=['Value'], i='name', j='Date').reset_index()\
        .sort_values(['location', 'name'])

print(res)

   name        Date location  Value
0  test    Jan-2010        A     12
2  test    Feb-2010        A     20
4  test  March-2010        A     30
1   foo    Jan-2010        B     18
3   foo    Feb-2010        B     20
5   foo  March-2010        B     25
jpp
fonte