Создание фрейма данных из словаря, где записи имеют разную длину

115

Скажем, у меня есть словарь с 10 парами ключ-значение. Каждая запись содержит массив numpy. Однако длина массива не у всех одинакова.

Как я могу создать фрейм данных, в котором каждый столбец содержит отдельную запись?

Когда я пытаюсь:

pd.DataFrame(my_dict)

Я получил:

ValueError: arrays must all be the same length

Любой способ преодолеть это? Я рад, что Pandas использует NaNэти столбцы для более коротких записей.

Джош
источник

Ответы:

133

В Python 3.x:

In [6]: d = dict( A = np.array([1,2]), B = np.array([1,2,3,4]) )

In [7]: pd.DataFrame(dict([ (k,pd.Series(v)) for k,v in d.items() ]))
Out[7]: 
    A  B
0   1  1
1   2  2
2 NaN  3
3 NaN  4

В Python 2.x:

заменить d.items()на d.iteritems().

Джефф
источник
Недавно я работал над той же проблемой, и это лучше, чем было у меня! Следует отметить, что заполнение NaN приведет к принуждению серии dtype к float64, что может быть проблематичным, если вам нужно выполнить целочисленную математику.
mattexx 02
ты всегда можешь задать вопрос - на них многие люди отвечают
Джефф
вам необходимо предоставить MVCE, как предлагается в комментариях
Джефф,
3
@germ, вы можете сначала импортировать серию или сделать что-то вроде pd.Series(...) (при условии, что import pandas as pdв разделе импорта)
Нима Мусави
5
Более компактная версия этого ответа:pd.DataFrame({k: pd.Series(l) for k, l in d.items()})
user553965 04
82

Вот простой способ сделать это:

In[20]: my_dict = dict( A = np.array([1,2]), B = np.array([1,2,3,4]) )
In[21]: df = pd.DataFrame.from_dict(my_dict, orient='index')
In[22]: df
Out[22]: 
   0  1   2   3
A  1  2 NaN NaN
B  1  2   3   4
In[23]: df.transpose()
Out[23]: 
    A  B
0   1  1
1   2  2
2 NaN  3
3 NaN  4
dezzan
источник
есть ли другие варианты индексации?
sAguinaga
@sAguinaga Да:, columnsно это уже значение по умолчанию. См. Документацию по pandas - pandas.DataFrame.from_dict
Мурмель
15

Ниже приведен способ привести в порядок ваш синтаксис, но по-прежнему делать то же самое, что и эти другие ответы:

>>> mydict = {'one': [1,2,3], 2: [4,5,6,7], 3: 8}

>>> dict_df = pd.DataFrame({ key:pd.Series(value) for key, value in mydict.items() })

>>> dict_df

   one  2    3
0  1.0  4  8.0
1  2.0  5  NaN
2  3.0  6  NaN
3  NaN  7  NaN

Аналогичный синтаксис существует и для списков:

>>> mylist = [ [1,2,3], [4,5], 6 ]

>>> list_df = pd.DataFrame([ pd.Series(value) for value in mylist ])

>>> list_df

     0    1    2
0  1.0  2.0  3.0
1  4.0  5.0  NaN
2  6.0  NaN  NaN

Другой синтаксис для списков:

>>> mylist = [ [1,2,3], [4,5], 6 ]

>>> list_df = pd.DataFrame({ i:pd.Series(value) for i, value in enumerate(mylist) })

>>> list_df

   0    1    2
0  1  4.0  6.0
1  2  5.0  NaN
2  3  NaN  NaN

Вам может дополнительно потребоваться транспонировать результат и / или изменить типы данных столбца (с плавающей запятой, целое число и т. Д.).

OrangeSherbet
источник
3

Хотя это не дает прямого ответа на вопрос ОП. Я обнаружил, что это отличное решение для моего случая, когда у меня были неравные массивы, и я хотел бы поделиться:

из документации pandas

In [31]: d = {'one' : Series([1., 2., 3.], index=['a', 'b', 'c']),
   ....:      'two' : Series([1., 2., 3., 4.], index=['a', 'b', 'c', 'd'])}
   ....: 

In [32]: df = DataFrame(d)

In [33]: df
Out[33]: 
   one  two
a    1    1
b    2    2
c    3    3
d  NaN    4
user2015487
источник
3

Вы также можете использовать pd.concatвместе axis=1со списком pd.Seriesобъектов:

import pandas as pd, numpy as np

d = {'A': np.array([1,2]), 'B': np.array([1,2,3,4])}

res = pd.concat([pd.Series(v, name=k) for k, v in d.items()], axis=1)

print(res)

     A  B
0  1.0  1
1  2.0  2
2  NaN  3
3  NaN  4
jpp
источник
2

Обе следующие строки работают отлично:

pd.DataFrame.from_dict(df, orient='index').transpose() #A

pd.DataFrame(dict([ (k,pd.Series(v)) for k,v in df.items() ])) #B (Better)

Но с% timeit на Jupyter у меня есть 4-кратное соотношение скорости для B и A, что весьма впечатляет, особенно при работе с огромным набором данных (в основном с большим количеством столбцов / функций).

Исмаил Хачими
источник
1

Если вы не хотите, чтобы он отображался, NaNи у вас есть две определенные длины, добавление «пробела» в каждую оставшуюся ячейку также будет работать.

import pandas

long = [6, 4, 7, 3]
short = [5, 6]

for n in range(len(long) - len(short)):
    short.append(' ')

df = pd.DataFrame({'A':long, 'B':short}]
# Make sure Excel file exists in the working directory
datatoexcel = pd.ExcelWriter('example1.xlsx',engine = 'xlsxwriter')
df.to_excel(datatoexcel,sheet_name = 'Sheet1')
datatoexcel.save()

   A  B
0  6  5
1  4  6
2  7   
3  3   

Если у вас более двух длин записей, рекомендуется создать функцию, которая использует аналогичный метод.

Рохан Чандратре
источник
-3

pd.DataFrame ([my_dict]) подойдет!

Джон Джой
источник
нет, если массивы в dict имеют разную длину
baxx