从单一列表构建字典的大多数Pythonic方法

时间:2013-01-10 13:25:59

标签: python list dictionary

我有一个日期名称列表(通常是周一至周六,但特殊情况适用),我想创建一个字典。我想将每天的值初始化为零。

如果我有一个零列表与天数列表相同的长度,这将是zip()的一个简单用例。但是,零列表是浪费空间,如果这是唯一的解决方案,我会尽快做出类似的事情:

for day in weekList:
    dayDict[day] = 0

有更多的pythonic方式吗?

2 个答案:

答案 0 :(得分:25)

使用.fromkeys() class method

dayDict = dict.fromkeys(weekList, 0)

它使用第一个参数(序列)中的元素作为键来构建字典,并使用第二个参数(默认为None)作为所有条目的值。

就其本质而言,此方法将重用所有键的值;不要传递一个可变值,如列表或字典,并期望它为每个键创建该可变值的单独副本。在这种情况下,请使用字典理解:

dayDict = {d: [] for d in weekList}

答案 1 :(得分:17)

dict.fromkeys外,您还可以使用dict-comprehension, 但fromkeys()比dict理解更快:

In [27]: lis = ['a', 'b', 'c', 'd']

In [28]: dic = {x: 0 for x in lis}

In [29]: dic
Out[29]: {'a': 0, 'b': 0, 'c': 0, 'd': 0}

对于2.6及更早版本:

In [30]: dic = dict((x, 0) for x in lis)

In [31]: dic
Out[31]: {'a': 0, 'b': 0, 'c': 0, 'd': 0}

timeit比较:

In [38]: %timeit dict.fromkeys(xrange(10000), 0)         # winner
1000 loops, best of 3: 1.4 ms per loop

In [39]: %timeit {x: 0 for x in xrange(10000)}
100 loops, best of 3: 2.08 ms per loop

In [40]: %timeit dict((x, 0) for x in xrange(10000))
100 loops, best of 3: 4.63 ms per loop

正如@Eumiro和@mgilson的评论中所提到的,重要的是要注意,如果使用的值是可变对象,fromkeys()dict-comprehensions可能会返回不同的对象:

In [42]: dic = dict.fromkeys(lis, [])

In [43]: [id(x) for x in dic.values()]
Out[43]: [165420716, 165420716, 165420716, 165420716] # all point to a same object

In [44]: dic = {x: [] for x in lis}

In [45]: [id(x) for x in dic.values()]
Out[45]: [165420780, 165420940, 163062700, 163948812]  # unique objects