在python中的字典的深拷贝

Deep copy of a dict in python

我想用python制作一个dict的深度拷贝。不幸的是,.deepcopy()方法不存在于dict中。我该怎么做?

1
2
3
4
5
6
7
8
9
>>> my_dict = {'a': [1, 2, 3], 'b': [4, 5, 6]}
>>> my_copy = my_dict.deepcopy()
Traceback (most recent calll last):
  File"<stdin>", line 1, in <module>
AttributeError: 'dict' object has no attribute 'deepcopy'
>>> my_copy = my_dict.copy()
>>> my_dict['a'][2] = 7
>>> my_copy['a'][2]
7

最后一行应该是3

我希望my_dict中的修改不会影响快照my_copy

我该怎么做?该解决方案应该与python 3.x兼容。


怎么样:

1
2
3
import copy
d = { ... }
d2 = copy.deepcopy(d)

Python 2或3:

1
2
3
4
5
6
7
8
9
Python 3.2 (r32:88445, Feb 20 2011, 21:30:00) [MSC v.1500 64 bit (AMD64)] on win32
Type"help","copyright","credits" or"license" for more information.
>>> import copy
>>> my_dict = {'a': [1, 2, 3], 'b': [4, 5, 6]}
>>> my_copy = copy.deepcopy(my_dict)
>>> my_dict['a'][2] = 7
>>> my_copy['a'][2]
3
>>>


dict.copy()是字典的一个浅拷贝函数id是一个内置函数,它提供变量的地址

首先,你需要理解"为什么会发生这个特殊的问题?"

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
In [1]: my_dict = {'a': [1, 2, 3], 'b': [4, 5, 6]}

In [2]: my_copy = my_dict.copy()

In [3]: id(my_dict)
Out[3]: 140190444167808

In [4]: id(my_copy)
Out[4]: 140190444170328

In [5]: id(my_copy['a'])
Out[5]: 140190444024104

In [6]: id(my_dict['a'])
Out[6]: 140190444024104

键"A"的两个dict中的列表地址指向同一位置。因此,当您更改"我的字典"中列表的值时,"我的副本"中的列表也会更改。

解决方案:

1
2
3
4
In [7]: my_copy = {key: value[:] for key, value in my_dict.items()}

In [8]: id(my_copy['a'])
Out[8]: 140190444024176

或者您可以使用上面提到的deepcopy。


Python 3 x

从复制导入deepcopy

1
2
my_dict = {'one': 1, 'two': 2}
new_dict_deepcopy = deepcopy(my_dict)

如果没有deepcopy,我无法从域字典中删除主机名字典。

如果没有deepcopy,我会得到以下错误:

1
"RuntimeError: dictionary changed size during iteration"

…当我试图从另一本字典中删除所需元素时。

1
2
3
import socket
import xml.etree.ElementTree as ET
from copy import deepcopy

域是字典对象

1
2
3
4
5
6
7
def remove_hostname(domain, hostname):
    domain_copy = deepcopy(domain)
    for domains, hosts in domain_copy.items():
        for host, port in hosts.items():
           if host == hostname:
                del domain[domains][host]
    return domain

实例输出:[orginal]域='localdomain':'localhost':'all':'4000'

[新]域='本地域':

所以这里所发生的是我在迭代一个字典的副本,而不是迭代字典本身。使用此方法,您可以根据需要删除元素。


一个更简单的(在我看来)解决方案是创建一个新字典并用旧字典的内容更新它:

1
2
3
4
5
6
7
8
9
10
11
12
13
my_dict={'a':1}

my_copy = {}

my_copy.update( my_dict )

my_dict['a']=2

my_dict['a']
Out[34]: 2

my_copy['a']
Out[35]: 1

这种方法的问题在于它可能不够"深入"。也就是说,不是递归深度。对于简单的对象来说已经足够好了,但对于嵌套字典来说就不够好了。下面是一个可能不够深的例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
my_dict1={'b':2}

my_dict2={'c':3}

my_dict3={ 'b': my_dict1, 'c':my_dict2 }

my_copy = {}

my_copy.update( my_dict3 )

my_dict1['b']='z'

my_copy
Out[42]: {'b': {'b': 'z'}, 'c': {'c': 3}}

通过使用deepcopy(),我可以消除半浅层的行为,但我认为必须决定哪种方法适合您的应用程序。在大多数情况下,你可能不在乎,但应该意识到可能的陷阱…最后的例子:

1
2
3
4
5
6
7
8
import copy

my_copy2 = copy.deepcopy( my_dict3 )

my_dict1['b']='99'

my_copy2
Out[46]: {'b': {'b': 'z'}, 'c': {'c': 3}}