如何将元组列表转换为字典

3

我有一个元组列表,像这样:

lst_of_tpls = [(1, 'test2', 3, 4),(11, 'test12', 13, 14),(21, 'test22', 23,24)]

我希望将其转换为字典,使其看起来像这样:

mykeys = ['ones', 'text', 'threes', 'fours']
mydict = {'ones': [1,11,21], 'text':['test2','test12','test22'], 
          'threes': [3,13,23], 'fours':[4,14,24]}

我尝试列举lst_of_tpls的方式如下:

mydict = dict.fromkeys(mykeys, [])
for count, (ones, text, threes, fours) in enumerate(lst_of_tpls):
    mydict['ones'].append(ones)

但是这会使我想要在“ones”中看到的值也出现在其他“类别”中:
{'ones': [1, 11, 21], 'text': [1, 11, 21], 'threes': [1, 11, 21], 'fours': [1, 11, 21]}

此外,我希望保持mykeys的灵活性。

@BearBrown 你不必同意吗?不确定你为什么要告诉我这个? - Torxed
2个回答

6
你可以两次使用zip功能来找到正确的配对:
lst_of_tpls = [(1, 'test2', 3, 4),(11, 'test12', 13, 14),(21, 'test22', 23,24)]
mykeys = ['ones', 'text', 'threes', 'fours']
new_d = {a:list(b) for a, b in zip(mykeys, zip(*lst_of_tpls))}

输出:

{
 'ones': [1, 11, 21],
 'text': ['test2', 'test12', 'test22'],
 'threes': [3, 13, 23],
 'fours': [4, 14, 24]
}

好的,很棒它可以工作。 对我来说,有很多新语法;让我们看看我是否理解它:zip(*lst_of_tpls) 返回一个迭代器,其中包含元组的第一个、第二个、... 元素。外部 zip 将键添加到其中。因此,现在我有一个键、元组对的迭代器。然后,我使用这个迭代器生成字典,并且动态转换第一个 zip 会给我的元组为列表。 - evilolive
@evilolive 正确。zip(*lst_of_tpls)lst_of_tpls中的元素配对,第二个zipmykeys中的每个元素与zip(*lst_of_tpls)中的每个元素配对。 - Ajax1234

1
你可以传递 (key, value) 元组到字典中,这比使用字典解析要快两倍。
lst_of_tpls = [(1, "test2", 3, 4), (11, "test12", 13, 14), (21, "test22", 23, 24)]
mykeys = ["ones", "text", "threes", "fours"]
my_dict = dict(zip(mykeys, zip(*lst_of_tpls)))

输出:

{'ones': (1, 11, 21),
 'text': ('test2', 'test12', 'test22'),
 'threes': (3, 13, 23),
 'fours': (4, 14, 24)}

分析器示例:

lst_of_tpls = [(1, "test2", 3, 4), (11, "test12", 13, 14), (21, "test22", 23, 24)]
mykeys = ["ones", "text", "threes", "fours"]


def dict_comprehension():
    return {a: list(b) for a, b in zip(mykeys, zip(*lst_of_tpls))}


def dict_generator():
    return dict(zip(mykeys, zip(*lst_of_tpls)))


if __name__ == "__main__":
    import timeit

    funcs = (dict_comprehension, dict_generator)
    for f in funcs:
        result = timeit.timeit(f, number=10000, globals=globals())
        print(f"{f.__name__}: {result:.5f}")


dict_comprehension: 0.05009 
dict_generator: 0.02468

有趣。你能否尝试解释一下字典推导和创建新字典(...)之间的区别是什么?这是一个普遍规则,也适用于[...]与列表(...)创建吗? - evilolive

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