Python"扩展"为字典

Fer*_*anB 417 python dictionary

哪个是用另一个字典扩展字典的最佳方法?例如:

>>> a = { "a" : 1, "b" : 2 }
>>> b = { "c" : 3, "d" : 4 }
>>> a
{'a': 1, 'b': 2}
>>> b
{'c': 3, 'd': 4}
Run Code Online (Sandbox Code Playgroud)

我正在寻找任何操作来获得这个避免for循环:

{ "a" : 1, "b" : 2, "c" : 3, "d" : 4 }
Run Code Online (Sandbox Code Playgroud)

我希望做的事情如下:

a.extend(b)  # This does not work
Run Code Online (Sandbox Code Playgroud)

Nic*_*cue 638

a.update(b)
Run Code Online (Sandbox Code Playgroud)

最新的Python标准库文档

  • 请记住,update()直接修改dict并返回None. (44认同)
  • @Nearoo - 只需更新相反的方式; 而不是x.update(y)[将用y覆盖x值],使用y.update(x)[用x覆盖y值]并使用y作为您选择的dict进行进一步操作 (15认同)
  • 在阅读完文档后,人们将理解为什么会有'更新'而不是'扩展'. (4认同)
  • 但是,如果您只想使用已经未定义的值扩展dict(即不覆盖现有值),该怎么办?例如,更新一个字典`{"a":2,"b":3}`用dict` {"b":4,"c":5}`来判断`{"a":2,"b ":3," C":5}`?当然可以通过移动一些东西来使用`update()`,但如果只能在一行中完成它会更好...... (4认同)
  • 请记住,“更新”会默默地覆盖现有条目。即,对于重叠键,“b”中的任何值都会覆盖“a”中的值。 (2认同)

Tom*_*eys 176

这个封闭的问题是一个美丽的宝石:

"oneliner方式",既不改变输入词,也是

basket = dict(basket_one, **basket_two)
Run Code Online (Sandbox Code Playgroud)

了解什么是**basket_two(在**)指这里.

如果发生冲突,项目basket_two将覆盖来自的项目basket_one.作为单行人员,这是非常可读和透明的,并且我没有任何强制反对使用它的任何时候一个混合的两个其他人派上用场(任何难以理解它的读者实际上将得到很好的服务这促使他或她学习dict**形式的方式;-).因此,例如,使用如下:

x = mungesomedict(dict(adict, **anotherdict))
Run Code Online (Sandbox Code Playgroud)

在我的代码中经常发生.

最初由Alex Martelli提交

注意:在Python 3中,只有当basket_two中的每个键都是a时才会有效string.

  • [`dict`](http://docs.python.org/2/library/stdtypes.html#dict)的文档很容易找到[`**`](http://www.saltycrane.com/ blog/2008/01/how-to-use-args-and-kwargs-in-python /)有点棘手(关键字是*kwargs*).这是一个很好的解释:http://www.saltycrane.com/blog/2008/01/how-to-use-args-and-kwargs-in-python/ (3认同)
  • 可以用一个命令来生成第二个变量,而正如名字所说的`basket_one.update(<dict>)`可以更新现有字典(或克隆的字典)。 (3认同)
  • 请注意,在 Python 3 中,参数名称以及 `**anotherdict` 的键必须是字符串。 (2认同)

Vla*_*den 33

您是否尝试过使用字典映射的字典理解:

c = {**a, **b}
# c = { "a" : 1, "b" : 2, "c" : 3, "d" : 4 }
Run Code Online (Sandbox Code Playgroud)

  • 此语法是 [相当新](/sf/ask/2729121/) (Python 3.5) (4认同)
  • 仅供参考:这不是Python 2.7中的有效语法 (2认同)

var*_*tec 24

a.update(b)
Run Code Online (Sandbox Code Playgroud)

b键和值添加到a,如果已经存在键的值,则覆盖.


Aar*_*_ab 22

请注意,自Python 3.9以来,引入了更简单的语法(联合运算符):

d1 = {'a': 1}
d2 = {'b': 2}

extended_dict = d1 | d2
>> {'a':1, 'b': 2}
Run Code Online (Sandbox Code Playgroud)

注意:如果第一个字典与第二个字典共享密钥,位置很重要!

d1 = {'b': 1}
d2 = {'b': 2}
d1 | d2 
>> {'b': 2} 
Run Code Online (Sandbox Code Playgroud)

相关政治公众人物


ebl*_*ume 17

正如其他人所提到的那样,a.update(b)对于一些决定而言a,b你将在你的问题中找到你所要求的结果.但是,我想指出,我已经多次看到extend映射/设置对象的方法希望在语法中a.extend(b),a值的值不应该被覆盖b.a.update(b)覆盖a的价值观,所以不是一个好的选择extend.

请注意,有些语言调用这个方法defaults还是inject,因为它可以被看作是注射B的值(这可能是一组默认值),在以字典的方式,而不会覆盖可能已经存在的值.

当然,你可以简单地注意到它a.extend(b)几乎相同b.update(a); a=b.要删除分配,您可以这样做:

def extend(a,b):
    """Create a new dictionary with a's properties extended by b,
    without overwriting.

    >>> extend({'a':1,'b':2},{'b':3,'c':4})
    {'a': 1, 'c': 4, 'b': 2}
    """
    return dict(b,**a)
Run Code Online (Sandbox Code Playgroud)

感谢Tom Leys使用无副作用dict构造函数的聪明想法extend.


Ivo*_*ers 5

您还可以使用Python 3.3中引入的Python Collections.ChainMap 。

from collections import ChainMap
c = ChainMap(a, b)
c['a'] # returns 1
Run Code Online (Sandbox Code Playgroud)

这有一些可能的优点,具体取决于您的用例。这里对它们进行了更详细的解释,但我将给出一个简短的概述:

  • 链图仅使用字典的视图,因此实际上不会复制任何数据。这会导致更快的链接(但查找速度更慢)
  • 实际上没有键被覆盖,因此,如有必要,您可以知道数据是来自 a 还是 b。

这主要使它对于配置字典之类的东西很有用。