将带有列表的嵌套字典展开到pandas DataFrame中

问题描述 投票:5回答:2

我有一个嵌套字典,子字典使用列表:

nested_dict = {'string1': {69: [1231, 232], 67:[682, 12], 65: [1, 1]}, 
    `string2` :{28672: [82, 23], 22736:[82, 93, 1102, 102], 19423: [64, 23]}, ... }

列表中至少有两个元素用于子词典,但可能还有更多。

我想将这个字典“展开”成一个pandas DataFrame,第一个字典键有一列(例如'string1','string2',..),一个列用于子目录键,一列用于第一个字典键列表中的项目,下一个项目的一列,依此类推。

这是输出应该是什么样子:

col1       col2    col3     col4    col5    col6
string1    69      1231     232
string1    67      682      12
string1    65      1        1
string2    28672   82       23
string2    22736   82       93      1102    102
string2    19423   64       23

当然,我尝试使用pd.DataFrame.from_dict

new_df = pd.DataFrame.from_dict({(i,j): nested_dict[i][j] 
                           for i in nested_dict.keys() 
                           for j in nested_dict[i].keys()
                           ... 

现在我被卡住了。并且存在许多问题:

  1. 如何解析字符串(即nested_dict[i].values()),使每个元素都是新的pandas DataFrame列?
  2. 以上实际上不会为每个字段创建一列
  3. 以上内容不会填充带有元素的列,例如: string1应该位于子目录键值对的每一行中。 (对于col5col6,我可以用零填充NA)
  4. 我不确定如何正确命名这些列。
python pandas dictionary
2个回答
1
投票

这应该会给你你想要的结果,虽然它可能不是最优雅的解决方案。这可能是更好的(更多pandas方式)。

我解析了你的嵌套字典并构建了一个字典列表(每行一个)。

# some sample input
nested_dict = {
    'string1': {69: [1231, 232], 67:[682, 12], 65: [1, 1]}, 
    'string2' :{28672: [82, 23], 22736:[82, 93, 1102, 102], 19423: [64, 23]},
    'string3' :{28673: [83, 24], 22737:[83, 94, 1103, 103], 19424: [65, 24]}
}

# new list is what we will use to hold each row
new_list = []
for k1 in nested_dict:
    curr_dict = nested_dict[k1]
    for k2 in curr_dict:
        new_dict = {'col1': k1, 'col2': k2}
        new_dict.update({'col%d'%(i+3): curr_dict[k2][i] for i in range(len(curr_dict[k2]))})
        new_list.append(new_dict)

# create a DataFrame from new list
df = pd.DataFrame(new_list)

输出:

      col1   col2  col3  col4    col5   col6
0  string2  28672    82    23     NaN    NaN
1  string2  22736    82    93  1102.0  102.0
2  string2  19423    64    23     NaN    NaN
3  string3  19424    65    24     NaN    NaN
4  string3  28673    83    24     NaN    NaN
5  string3  22737    83    94  1103.0  103.0
6  string1     65     1     1     NaN    NaN
7  string1     67   682    12     NaN    NaN
8  string1     69  1231   232     NaN    NaN

假设输入总是包含足够的数据来创建col1col2

我循环通过nested_dict。假设nested_dict的每个元素也是字典。我们也循环通过那个字典(curr_dict)。密钥k1k2用于填充col1col2。对于其余的键,我们遍历列表内容并为每个元素添加一列。


2
投票

这是一个使用递归生成器展开嵌套字典的方法。它不会假设您有两个级别,但继续展开每个dict直到它击中list

nested_dict = {
    'string1': {69: [1231, 232], 67:[682, 12], 65: [1, 1]}, 
    'string2' :{28672: [82, 23], 22736:[82, 93, 1102, 102], 19423: [64, 23]},
    'string3': [101, 102]}

def unroll(data):
    if isinstance(data, dict):
        for key, value in data.items():
            # Recursively unroll the next level and prepend the key to each row.
            for row in unroll(value):
                yield [key] + row
    if isinstance(data, list):
        # This is the bottom of the structure (defines exactly one row).
        yield data

df = pd.DataFrame(list(unroll(nested_dict)))

因为unroll生成列表而不是dicts,所以列将以数字命名(在这种情况下从0到5)。所以你需要使用rename来获得你想要的列标签:

df.rename(columns=lambda i: 'col{}'.format(i+1))

这将返回以下结果(请注意,另外的string3条目也将展开)。

      col1   col2  col3   col4    col5   col6
0  string1     69  1231  232.0     NaN    NaN
1  string1     67   682   12.0     NaN    NaN
2  string1     65     1    1.0     NaN    NaN
3  string2  28672    82   23.0     NaN    NaN
4  string2  22736    82   93.0  1102.0  102.0
5  string2  19423    64   23.0     NaN    NaN
6  string3    101   102    NaN     NaN    NaN
© www.soinside.com 2019 - 2024. All rights reserved.