在词典词典中总结值

时间:2018-02-27 13:54:29

标签: python python-3.x dictionary sum

我有这样的事情:

> d1 = {'System tests': {'failed': 5, 'passed': 0, 'total': 5},
       'Func tests': {'failed': 5, 'passed': 0, 'total': 5}}

> d2 = {'System tests': {'failed': 1, 'passed': 1, 'total': 2}, 
        'Func tests': {'failed': 3, 'passed': 2, 'total': 5}}

> d3 = {'System tests': {'failed': 0, 'passed': 0, 'total': 0}, 
        'Func tests': {'failed': 1, 'passed': 0, 'total': 1}}

我想将值'失败',传递和总数合并到一个词典

所以输出应该是这样的:

d4 = {'System tests': {'failed': 6, 'passed': 1, 'total': 7}, 
       'Func tests': {'failed': 9, 'passed': 2, 'total': 11}

做这种事最简单的解决办法是什么?

我可以使用除集合之外的基本库。

解决方案必须是通用的,例如,如果将来出现其他一些词典

5 个答案:

答案 0 :(得分:5)

输入:

d1 = {'a': 100, 'b': 200, 'c':300}
d2 = {'a': 300, 'b': 200, 'd':400}
d = {k : d1.get(k, 0) + d2.get(k,0) for k in set(d1.keys()) | set(d2.keys())}

输出:

{'a': 400, 'b': 400, 'c': 300, 'd': 400}

答案 1 :(得分:3)

使用Counterdefaultdict非常简单,如:

代码:

d4 = defaultdict(Counter)
for d in d1, d2, d3:
    for k, subd in d.items():
        d4[k].update(subd)

测试代码:

d1 = {'System tests': {'failed': 5, 'passed': 0, 'total': 5},
        'Func tests': {'failed': 5, 'passed': 0, 'total': 5}}

d2 = {'System tests': {'failed': 1, 'passed': 1, 'total': 2},
        'Func tests': {'failed': 3, 'passed': 2, 'total': 5}}

d3 = {'System tests': {'failed': 0, 'passed': 0, 'total': 0},
        'Func tests': {'failed': 1, 'passed': 0, 'total': 1}}

from collections import Counter, defaultdict

d4 = defaultdict(Counter)
for d in d1, d2, d3:
    for k, subd in d.items():
        d4[k].update(subd)
print(d4)

结果:

defaultdict(<class 'collections.Counter'>, {
    'System tests': Counter({'total': 7, 'failed': 6, 'passed': 1}), 
    'Func tests': Counter({'total': 11, 'failed': 9, 'passed': 2})
})

答案 2 :(得分:1)

这将为您提供您正在寻找的输出,没有库。

d4 = {}

for d in d1, d2, d3:
    for test, results in d.items():
        if test not in d4:
            d4[test] = {}
        for key, value in results.items():
            if key in d4[test]:
                d4[test][key] += value
            else:
                d4[test][key] = value

结果:

{'System tests': {'failed': 6, 'passed': 1, 'total': 7}, 'Func tests': {'failed': 9, 'passed': 2, 'total': 11}}

答案 3 :(得分:0)

如果您无法使用collections.Counter,则另一种解决方案是使用reduce

from functools import reduce # reduce was moved here in Python3

def add_dicts(dict_list):

    def add(fst, snd):
        return {k: fst[k] + snd[k] for k in fst}

    return reduce(add, dict_list[1:], dict_list[0])

在代码中使用此代码将如下所示。

dict_list = [d1, d2, d3]

d4 = {}

for k in dict_list[0]:
    d4[k] = add_dicts([d[k] for d in dict_list])

尽管如此,这假设您的所有dict都已正确格式化。

答案 4 :(得分:0)

您可以将zipdict.items一起使用:

d1 = {'System tests': {'failed': 5, 'passed': 0, 'total': 5},
   'Func tests': {'failed': 5, 'passed': 0, 'total': 5}}
d2 = {'System tests': {'failed': 1, 'passed': 1, 'total': 2}, 
    'Func tests': {'failed': 3, 'passed': 2, 'total': 5}}
d3 = {'System tests': {'failed': 0, 'passed': 0, 'total': 0}, 
    'Func tests': {'failed': 1, 'passed': 0, 'total': 1}}

def combine_dicts(*d):
   return {a:{i:sum(h[i] for h in [b, c, d]) for i in ['failed', 'passed', 'total']} for [a, b], [_, c], [_, d] in zip(*d)}

print(combine_dicts(d1.items(), d2.items(), d3.items()))

输出:

{'System tests': {'failed': 6, 'total': 7, 'passed': 1}, 'Func tests': {'failed': 9, 'total': 11, 'passed': 2}}