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 同意!但是您还应该注意到,列表切片不适用于 dict value[:]。解决方案是针对问题中提到的特定数据结构,而不是通用解决方案。 字典理解的第二个解决方案不适用于嵌套列表作为值``` >>> 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的深拷贝的主要内容,如果未能解决你的问题,请参考以下文章

python的深拷贝与浅拷贝

清晰python的深拷贝和浅拷贝

清晰python的深拷贝和浅拷贝

**Python中的深拷贝和浅拷贝详解

Python中的深拷贝和浅拷贝

Python中的深拷贝与浅拷贝