三个列表合并为字典列表

20

考虑以下内容:

>>> # list of length n
>>> idx = ['a', 'b', 'c', 'd']

>>> # list of length n
>>> l_1 = [1, 2, 3, 4]

>>> # list of length n
>>> l_2 = [5, 6, 7, 8]

>>> # first key
>>> key_1 = 'mkt_o'

>>> # second key
>>> key_2 = 'mkt_c'

如何将这个杂乱无章的东西压缩成这样?

{
    'a': {'mkt_o': 1, 'mkt_c': 5},
    'b': {'mkt_o': 2, 'mkt_c': 6},
    'c': {'mkt_o': 3, 'mkt_c': 6},
    'd': {'mkt_o': 4, 'mkt_c': 7},
    ...
}

我最接近的是像这样的东西:

>>> dict(zip(idx, zip(l_1, l_2)))
{'a': (1, 5), 'b': (2, 6), 'c': (3, 7), 'd': (4, 8)}

当然,它的值是元组而不是字典。

>>> dict(zip(('mkt_o', 'mkt_c'), (1,2)))
{'mkt_o': 1, 'mkt_c': 2}

这似乎很有前途,但是再次失败了,未能满足要求。

4个回答

41
{k : {key_1 : v1, key_2 : v2} for k,v1,v2 in zip(idx, l_1, l_2)}

1
请鞠躬致意,观众正在欢呼。完美,谢谢。 - Jason Strimpel
1
这是最快的方法:>>> sol_1 = '{k: dict(zip(["mkt_o", "mkt_c"], v)) for k, v in zip(["a", "b", "c", "d"], zip([1, 2, 3, 4], [5, 6, 7, 8]))}' >>> sol_2 = 'dict(zip(["a", "b", "c", "d"], [{"mkt_o": i, "mkt_c": j} for i, j in zip([1, 2, 3, 4], [5, 6, 7, 8])]))' >>> sol_3 = '{k : {"mkt_o" : v1, "mkt_c" : v2} for k,v1,v2 in zip(["a", "b", "c", "d"], [1, 2, 3, 4], [5, 6, 7, 8])}' >>> timeit.timeit(sol_1, number=10000) 0.05262671899981797 >>> timeit.timeit(sol_2, number=10000) 0.02170446099989931 >>> timeit.timeit(sol_3, number=10000) 0.014590603001124691 - Jason Strimpel

15

解决方法 1:您可以使用zip两次(实际上是三次),结合字典推导式以实现此操作:

idx = ['a', 'b', 'c', 'd']
l_1 = [1, 2, 3, 4]
l_2 = [5, 6, 7, 8]

keys = ['mkt_o', 'mkt_c']   # yours keys in another list

new_dict = {k: dict(zip(keys, v)) for k, v in zip(idx, zip(l_1, l_2))}

解决方案2: 您还可以使用嵌套的列表推导式zip,如下所示:

Solution 2: 您还可以使用嵌套的列表推导式zip,如下所示:

new_dict = dict(zip(idx, [{key_1: i, key_2: j} for i, j in zip(l_1, l_2)]))

解决方案3:使用字典推导式zip的基础上,如DYZ的答案所示:

new_dict = {k : {key_1 : v1, key_2 : v2} for k,v1,v2 in zip(idx, l_1, l_2)}

以上所有的解决方案都将返回new_dict

{
     'a': {'mkt_o': 1, 'mkt_c': 5}, 
     'b': {'mkt_o': 2, 'mkt_c': 6}, 
     'c': {'mkt_o': 3, 'mkt_c': 7},
     'd': {'mkt_o': 4, 'mkt_c': 8}
 }

1
第一种解决方案的优点是可以处理“n”个键。不错! - Eric Duminil

2

如果你正在使用字典、列表、索引和键,并希望将数据转置,建议使用pandasDataFrame.T.to_dict):

>>> import pandas as pd
>>> idx = ['a', 'b', 'c', 'd']
>>> l_1 = [1, 2, 3, 4]
>>> l_2 = [5, 6, 7, 8]
>>> key_1 = 'mkt_o'
>>> key_2 = 'mkt_c'
>>> pd.DataFrame([l_1, l_2], index=[key_1, key_2], columns = idx)
       a  b  c  d
mkt_o  1  2  3  4
mkt_c  5  6  7  8
>>> pd.DataFrame([l_1, l_2], index=[key_1, key_2], columns = idx).T
   mkt_o  mkt_c
a      1      5
b      2      6
c      3      7
d      4      8
>>> pd.DataFrame([l_1, l_2], index=[key_1, key_2], columns = idx).to_dict()
{'a': {'mkt_o': 1, 'mkt_c': 5},
 'b': {'mkt_o': 2, 'mkt_c': 6},
 'c': {'mkt_o': 3, 'mkt_c': 7},
 'd': {'mkt_o': 4, 'mkt_c': 8}
}

1

这也可以使用dict、zip、map和itertools中的repeat来完成:

>>> from itertools import repeat
>>> dict(zip(idx, map(dict, zip(zip(repeat(key_1), l_1), zip(repeat(key_2), l_2)))))
{'a': {'mkt_c': 5, 'mkt_o': 1}, 'c': {'mkt_c': 7, 'mkt_o': 3}, 'b': {'mkt_c': 6, 'mkt_o': 2}, 'd': {'mkt_c': 8, 'mkt_o': 4}}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接