python - 根据匹配的键:值对组合字典列表中的字典
问题描述
我想在具有匹配键:值对的字典列表中组合/合并多个字典。但是,我不希望这些对中的“值”本身成为每个新形成的字典中的列表。每个字典都有相同的键集,但大多数键:值对将是唯一的,除了我想要组合的那个。我知道我可以根据匹配键组合字典,例如在本例中:
>>>from collections import defaultdict
d1 = {'A': 'str1', 'B': 41, 'C': 32}
d2 = {'A': 'str1', 'B': 12, 'C': 62}
d3 = {'A': 'str2', 'B': 47, 'C': 73}
d4 = {'A': 'str2', 'B': 101, 'C': 93}
dd = defaultdict(list)
for d in [d1, d2, d3, d4]: # you can list as many input dicts as you want here, I'll have a few thousand
for key, value in d.items():
dd[key].append(value)
print(dd)
这给了我预期的结果:
defaultdict(<class 'list'>, {'A': ['str1', 'str1', 'str2', 'str2'],
'B': [41, 12, 47, 101],
'C': [32, 62, 73, 93]})
当我想要的结果实际上是这样时:
[{'A': 'str1', 'B': [41, 12], 'C': [32, 62]},
{'A': 'str2', 'B': [47, 101], 'C': [73, 93]}]
例如,来自匹配键'B'
和'C'
形成列表的值,仅当键下存在匹配的键:值对时才组合字典,'A'
而不形成相同值的列表'A'
解决方案
您可以执行以下操作:
from collections import defaultdict
from functools import reduce
from itertools import chain
def merge(d1, d2, key='A'):
r = defaultdict(list)
for k, v in chain(d1.items(), d2.items()):
if k != key:
r[k].extend(v if isinstance(v, list) else [v])
return {**r, key: d1[key]}
d1 = {'A': 'str1', 'B': 41, 'C': 32}
d2 = {'A': 'str1', 'B': 12, 'C': 62}
d3 = {'A': 'str2', 'B': 47, 'C': 73}
d4 = {'A': 'str2', 'B': 101, 'C': 93}
common = defaultdict(list)
for d in [d1, d2, d3, d4]:
common[d['A']].append(d)
result = [reduce(merge, value) for value in common.values()]
print(result)
输出
[{'B': [41, 12], 'C': [32, 62], 'A': 'str1'}, {'B': [47, 101], 'C': [73, 93], 'A': 'str2'}]
推荐阅读
- r - 如何使用 ggplot2 在 R 中绘制多个响应调查项目?
- c - 机器码是如何从汇编中生成的?
- java - 我们应该如何在不使用 @XMLRootElement 的情况下配置响应对象
- ios - 如何快速将数据从第二个 VC 传递到第一个 VC 中容器内的 VC
- javascript - 如何使用 a 标签指向另一页上图像幻灯片中的图片
- activemq-artemis - 嵌入式 ActiveMQ Artemis 不支持管理 (getQueueNames)
- ios - 发布到 App Store 时应用加载程序挂起
- cmake - 如何使用 cmake 生成 .out 文件?
- excel - Sumproduct sum 与单个单元格有效,但不适用于多个单元格
- android - 缺少 AndroidManifest.xml 尝试重新导入插件