Como faço para mesclar uma lista de dicts em um único dict?

124

Como posso virar uma lista de dictos como este

[{'a':1}, {'b':2}, {'c':1}, {'d':2}]

Em um único ditado como este

{'a':1, 'b':2, 'c':1, 'd':2}
killown
fonte
É verdade, embora isso se funda em um dicionário de listas.
Katriel de
1
é uma lista e não um dict: >>> type ([{'a': 1}, {'b': 2}]) <type 'list'>
killown
Realmente deveria haver um forro para isso, considerando quantas vezes ele aparece. Existe uma razão pela qual {** d for d in d_list} não é compatível?
markemus de

Respostas:

163

Isso funciona para dicionários de qualquer tamanho:

>>> result = {}
>>> for d in L:
...    result.update(d)
... 
>>> result
{'a':1,'c':1,'b':2,'d':2}

Como uma compreensão :

# Python >= 2.7
{k: v for d in L for k, v in d.items()}

# Python < 2.7
dict(pair for d in L for pair in d.items())
wim
fonte
6
>>> L=[{'a': 1}, {'b': 2}, {'c': 1}, {'d': 2}]    
>>> dict(i.items()[0] for i in L)
{'a': 1, 'c': 1, 'b': 2, 'd': 2}

Nota: a ordem de 'b' e 'c' não corresponde à sua saída porque os dictos não estão ordenados

se os dicts podem ter mais de uma chave / valor

>>> dict(j for i in L for j in i.items())
John La Rooy
fonte
Falha se algum dos dictos da lista tiver mais de um item.
PaulMcG
2
tosse * (presumindo que cada dicionário contenha apenas um par de valores-chave) * tosse
Katriel
Bem, primeiro eu postei um método que funciona com dictos gerais, mas parecia que o OP não precisava disso
John La Rooy
5

Para dicionários planos, você pode fazer isso:

from functools import reduce
reduce(lambda a, b: dict(a, **b), list_of_dicts)
dietbuddha
fonte
4

É semelhante a @delnan, mas oferece a opção de modificar os itens k / v (chave / valor) e acredito que seja mais legível:

new_dict = {k:v for list_item in list_of_dicts for (k,v) in list_item.items()}

por exemplo, substitua k / v elems da seguinte forma:

new_dict = {str(k).replace(" ","_"):v for list_item in list_of_dicts for (k,v) in list_item.items()}

descompacta a tupla k, v do gerador de dicionário .items () depois de retirar o objeto dict da lista

Schalton
fonte
2
dict1.update( dict2 )

Isso é assimétrico porque você precisa escolher o que fazer com as chaves duplicadas; neste caso, dict2substituirádict1 . Troque-os pelo outro caminho.

EDIT: Ah, desculpe, não vi isso.

É possível fazer isso em uma única expressão:

>>> from itertools import chain
>>> dict( chain( *map( dict.items, theDicts ) ) )
{'a': 1, 'c': 1, 'b': 2, 'd': 2}

Nenhum crédito para mim por isso!

No entanto, eu diria que pode ser mais pitônico (explícito> implícito, plano> aninhado) fazer isso com um forloop simples . YMMV.

Katriel
fonte
1
é uma lista e não um dict: >>> type ([{'a': 1}, {'b': 2}]) <type 'list'>
killown
1

Você pode usar juntar a função de funcy biblioteca:

from funcy import join
join(list_of_dicts)
Suor
fonte
1

Uma pequena melhoria na resposta @dietbuddha com a descompactação do dicionário do PEP 448 , para mim fica mais legível dessa forma, também, é mais rápido também :

from functools import reduce
result_dict = reduce(lambda a, b: {**a, **b}, list_of_dicts)

Mas tenha em mente que isso funciona apenas com as versões Python 3.5+.

Insomniac631
fonte
0
>>> dictlist = [{'a':1},{'b':2},{'c':1},{'d':2, 'e':3}]
>>> dict(kv for d in dictlist for kv in d.iteritems())
{'a': 1, 'c': 1, 'b': 2, 'e': 3, 'd': 2}
>>>

Observação Eu adicionei um segundo par de chave / valor ao último dicionário para mostrar que ele funciona com várias entradas. Além disso, as chaves de dicts posteriores na lista sobrescreverão a mesma chave de um dict anterior.

Dave Kirby
fonte
-1

dic1 = {'Maria': 12, 'Paco': 22, 'José': 23} dic2 = {'Patrícia': 25, 'Marcos': 22 'Tomas': 36}

dic2 = dict (dic1.items () + dic2.items ())

e este será o resultado:

dic2 {'Jose': 23, 'Marcos': 22, 'Patricia': 25, 'Tomas': 36, 'Paco': 22, 'Maria': 12}

Max Herrera
fonte