问题
现在有多个字典或者映射,需要将它们从逻辑上合并为一个单一的映射后执行某些操作, 比如查找值或者检查某些键是否存在。
解决方案
假如你有如下两个字典:
a = {'x': 1, 'z': 3 }
b = {'y': 2, 'z': 4 }
现在假设必须在两个字典中执行查找操作,比如:先从 a 中找,如果找不到再在 b 中找。 一个非常简单的解决方案就是使用 collections 模块中的 ChainMap
类。比如:
from collections import ChainMap
c = ChainMap(a, b)
print(c['x']) # Outputs 1 (from a)
print(c['y']) # Outputs 2 (from b)
print(c['z']) # Outputs 3 (from a)
1
2
3
讨论
一个 ChainMap
接受多个字典并将它们在逻辑上变为一个字典。 但是,这些字典并不是真的合并在一起了, ChainMap 类只是在内部创建了一个容纳这些字典的列表,并重新定义了一些常见的字典操作来遍历这个列表。大部分字典操作都是可以正常使用的,比如:
print(len(c))
print(list(c.keys()))
print(list(c.values()))
3
['x', 'z', 'y']
[1, 3, 2]
如果出现重复键,那么第一次出现的映射值会被返回。 因此,示例程序中的 c['z']
总是会返回字典 a 中对应的值,而不是 b 中对应的值。
对于字典的更新或删除操作总是影响的是列表中第一个字典。比如:
c['z'] = 40
c['w'] = 100
del c['x']
print(a)
{'z': 40, 'w': 100}
del c['y']
Traceback (most recent call last):
...
KeyError: "Key not found in the first mapping: 'y'"
作为 ChainMap 的替代,你可能会考虑使用 update() 方法将两个字典合并。比如:
a = {'x': 1, 'z': 3 }
b = {'y': 2, 'z': 4 }
merge = dict(b)
merge.update(a)
print(merge)
{'y': 2, 'z': 3, 'x': 1}
这样也能行得通,但是它需要你创建一个完全不同的字典对象,或者是破坏现有字典结构。 同时,如果原字典做了更新,这种改变不会反应到新的合并字典中去。比如:
a['x'] = 100
print(merge['x'])
1
ChainMap
使用原来的字典,它自己不创建新的字典。所以它并不会产生上面所说的结果。
网友评论