Python在字典中将键映射到多个值上

Python在字典中将键映射到多个值上

问题

我们想要一个能将键(key)能映射到多个值的字典(即所谓的一键多值字典multidict)。

解决方案

字典是一种关联容器。每个键都映射到单独的值上。如果想让键映射到多个值,需要将这多个值保存到另一个容器如列表或集合中。例如,可能会像这样创建字典:

d = {
    'a' : [1, 2, 3],
    'b' : [4, 5]
}
e = {
    'a' : {1, 2, 3},
    'b' : {4, 5}
}

要使用列表还是集合完全取决于应用的意图。如果希望保留元素插入的顺序,就用列表。如果希望消除重复元素(且不在意它们的顺序),就用集合。
为了能方便的创建这样的字典,可以利用collections模块中的defaultdict类。defaultdict的一个特点就是它会自动初始化第一个值,这样只需关注添加元素即可。例如:

from collections import defaultdict
d = defaultdict(list)
d['a'].append(1)
d['a'].append(2)
d['a'].append(3)
d['b'].append(4)
d['b'].append(5)
print(d) # outputs : defaultdict(<class 'list'>, {'a': [1, 2, 3], 'b': [4, 5]})
d = defaultdict(set)
d['a'].add(1)
d['a'].add(2)
d['a'].add(3)
d['b'].add(4)
d['b'].add(5)
print(d) # outputs : defaultdict(<class 'set'>, {'a': {1, 2, 3}, 'b': {4, 5}})

关于defaultdict,需要注意的一个地方是,它会自动创建字典表项以待稍后的访问(即使这些表项当前在字典中还没有找到),例如:

from collections import defaultdict
d = defaultdict(list)
if d['a']:
    pass
print(d) # outputs : defaultdict(<class 'list'>, {'a': []})

如果不想要这个功能,可以在普通的字典上调用setdefault()方法来取代。例如:

from collections import defaultdict
d = {}
d.setdefault('a', []).append(1)
d.setdefault('a', []).append(2)
d.setdefault('a', []).append(3)
d.setdefault('b', []).append(4)
d.setdefault('b', []).append(5)
print(d) # outputs : {'a': [1, 2, 3], 'b': [4, 5]}

然后,总感觉使用setdefault()有点不自然,更别提每次调用它时都会创建一个初始值的新实例了(例子中的空列表[])。

讨论

原则上,构建一个一键多值字典是很容易的。但是试着自己对第一个值做初始化操作,这就会变得很杂乱。例如,可能会写下这样的代码:

d = {}
for key, value in pairs:
    if key not in d:
        d[key] = value
    d[key].append(value)

使用defaultdict后代码会清晰很多:

d = defaultdict(list)
for key, value in pairs:
    d[key].append(value)

原文地址

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 121,073评论 16 134
  • 第1章 准备工作第2章 Python语法基础,IPython和Jupyter第3章 Python的数据结构、函数和...
    SeanCheney阅读 97,339评论 25 272
  • 暗夜中,手机亮了。睁着朦胧的睡眼,陌看着枕边的手机。谁啊,现在给我发信息,好不容易才有时间休息的陌,困得不愿动。 ...
    屿上屿阅读 157评论 4 7
  • 秋收时节到了,那一年,我六岁,还没有上学。父母去地里掰玉米,嘱咐我在家看门,可以在家玩,也可以锁上门去院子外边,但...
    寒山红枫阅读 322评论 0 1
  • 写给X小姐,不是董小姐,本想趁着一股热乎劲儿强势装逼,写一篇自己满意的短文,不料灵感充足时放不下手上一堆工作,再次...
    隔家的小王阅读 114评论 0 0