python中dict的深拷贝
Posted
技术标签:
【中文标题】python中dict的深拷贝【英文标题】:Deep copy of a dict in python 【发布时间】:2011-07-03 14:15:45 【问题描述】:我想在 python 中制作dict
的深层副本。不幸的是,.deepcopy()
方法不存在 dict
。我该怎么做?
>>> 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 兼容。
【问题讨论】:
我不知道它是否重复,但这个:***.com/questions/838642/python-dictionary-deepcopy 非常接近。 【参考方案1】:怎么样:
import copy
d = ...
d2 = copy.deepcopy(d)
Python 2 或 3:
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
>>>
【讨论】:
确实适用于我给出的过于简单的示例。我的钥匙不是数字而是物体。如果我阅读了复制模块文档,我必须为键声明一个 __copy__()/__deepcopy__() 方法。非常感谢你带我去那里! Python 3.2 和 2.7 的代码有什么区别吗?他们似乎和我一模一样。如果是这样,最好是单个代码块和一条语句“适用于 Python 3 和 2” 还值得一提的是copy.deepcopy
不是线程安全的。学到了这个很难。另一方面,根据您的用例,json.loads(json.dumps(d))
是线程安全的,并且运行良好。
@rob 您应该发布该评论作为答案。这是一个可行的替代方案。线程安全的细微差别是一个重要的区别。
@BuvinJ 问题是json.loads
并不能解决python dict
属性不是JSON 可序列化的所有用例的问题。它可能会帮助那些只处理简单数据结构的人,例如来自 API,但我认为这不足以完全回答 OP 的问题。【参考方案2】:
dict.copy() 是字典的浅拷贝函数 id 是内置函数,可以为您提供变量的地址
首先您需要了解“为什么会出现这个特定问题?”
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
两个 dicts 中键“a”的列表地址指向相同的位置。 因此,当您更改 my_dict 中列表的值时,my_copy 中的列表也会更改。
问题中提到的数据结构解决方案:
In [7]: my_copy = key: value[:] for key, value in my_dict.items()
In [8]: id(my_copy['a'])
Out[8]: 140190444024176
或者你可以使用上面提到的 deepcopy。
【讨论】:
您的解决方案不适用于嵌套字典。出于这个原因,deepcopy 更可取。 @CharlesPlager 同意!但是您还应该注意到,列表切片不适用于 dictvalue[:]
。解决方案是针对问题中提到的特定数据结构,而不是通用解决方案。
字典理解的第二个解决方案不适用于嵌套列表作为值``` >>> my_copy = k:v.copy() for k,v in my_dict.items() >> > id(my_copy["a"]) 4427452032 >>> id(my_dict["a"]) 4427460288 >>> id(my_copy["a"][0]) 4427469056 >>> id(my_dict["a" ][0]) 4427469056 ```【参考方案3】:
Python 3.x
从复制导入 deepcopy
my_dict = 'one': 1, 'two': 2
new_dict_deepcopy = deepcopy(my_dict)
没有 deepcopy,我无法从我的域字典中删除主机名字典。
如果没有 deepcopy,我会收到以下错误:
"RuntimeError: dictionary changed size during iteration"
...当我尝试从另一个字典中的字典中删除所需元素时。
import socket
import xml.etree.ElementTree as ET
from copy import deepcopy
域是一个字典对象
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
示例输出: [原始]域 = 'localdomain': 'localhost': 'all': '4000'
[新]域 = 'localdomain':
所以这里发生的事情是我正在迭代字典的副本,而不是迭代字典本身。使用此方法,您可以根据需要删除元素。
【讨论】:
以上是关于python中dict的深拷贝的主要内容,如果未能解决你的问题,请参考以下文章