Python - 拷贝 - 浅拷贝(Shallow Copy)和深拷贝(Deep Copy)

文章详细解释了Python中浅拷贝(ShallowCopy)和深拷贝(DeepCopy)的区别,通过例子展示了如何在可变类型如列表中产生拷贝以及它们的修改影响。当使用`[0]*5`创建二维数组时,由于列表的可变性,所有行实际上共享同一个引用,导致修改一个元素会影响所有行。文章建议使用`copy`模块的`copy()`和`deepcopy()`函数来更精确地控制拷贝行为。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Python - 拷贝 - 浅拷贝(Shallow Copy)和深拷贝(Deep Copy)

前言

假设我以这样的方式创建一个3 x 5的二维数组:

a = [[0] * 5] * 3

然后我修改 a [ 2 ] [ 3 ] a[2][3] a[2][3]的值为 1 1 1

a[2][3] = 1

结果会发现数组 a a a中第二维坐标为 3 3 3的数全部被修改为了 1 1 1,而没有发生“第一维坐标为 2 2 2的数全部被改成了 1 1 1

print(a)  # [[0, 0, 0, 1, 0], [0, 0, 0, 1, 0], [0, 0, 0, 1, 0]]

原因

这就涉及到了Python中的拷贝机制。

Python中的数据按照其是否可以更改,可以分为两类:

  • 可变类型包括列表(list)、字典(dict)和集合(set)
  • 不可变类型包括整数(int)、浮点数(float)、布尔值(bool)、元组(tuple)和字符串(str)

深拷贝: 对于不可变类型(例如整数)进行复制操作时,会产生一个新的对象。对新对象的更改不会对旧对象造成影响:

a = 2
b = a
b = 1
print(a, b)  # 2 1
print(id(a), id(b))  # 2474931349840 2474931349808  # 不同

浅拷贝: 然而对于可变类型(例如列表)进行复制时,只会将对象的引用复制一份,它们实际指向同意对象。因此修改新的对象会对旧对象产生影响:

a = [1, 2, 3]
b = a
b[2] = 0
print(a, b)  # [1, 2, 0] [1, 2, 0]
print(id(a), id(b))  # 2537310019904 2537310019904  # 相同

注意对新对象的修改是指修改对象中的一部分,而不是让新对象指向另一个对象

a = [1, 2, 3]
b = a
b[0] = 0  # 修改对象中的一部分,这时a = [0, 2, 3]
b = [0]  # b指向了一个新的对象,原来的对象并没有被修改,这时a = [0, 2, 3]

这就解释了前言中的问题

[ 0 ] ∗ 5 [0] * 5 [0]5是将 0 0 0复制为5份, 0 0 0是不可变的整数,因此新列表 [ 0 , 0 , 0 , 0 , 0 ] [0, 0, 0, 0, 0] [0,0,0,0,0]中的每个 0 0 0都是独立的,修改其中一个 0 0 0不会影响到其他 0 0 0的值

但是 [ [ 0 , 0 , 0 , 0 , 0 ] ∗ 3 ] [[0, 0, 0, 0, 0] * 3] [[0,0,0,0,0]3]是将 [ 0 , 0 , 0 , 0 , 0 ] [0, 0, 0, 0, 0] [0,0,0,0,0]复制为5份, [ 0 , 0 , 0 , 0 , 0 ] [0, 0, 0, 0, 0] [0,0,0,0,0]是可变的列表,因此实质上是创建了 3 3 3个指向 [ 0 , 0 , 0 , 0 , 0 ] [0, 0, 0, 0, 0] [0,0,0,0,0]的对象,因此修改其中一个,另外两个也会随之变化。

但是:

a = [0] * 5
for i in range(5):
    print(id(a[i]), end=' ')
# 2977374300432 2977374300432 2977374300432 2977374300432 2977374300432  # 完全相同!!!
print(id(a[0] == id(a[1])))  # True
a[0] = 1
print(a)  # [1, 0, 0, 0, 0]
print(id(a[0]))  # 2977374300464
print(id(a[1]))  # 2977374300432
print(id(a[2]))  # 2977374300432
print(id(a[0]) == id(a[1]))  # False

也许是Py的优化?只有当修改不可变元素时才真的深拷贝?

TODO: import copy

可以研究一下 copy.copy()和copy.deepcopy()

原创不易,转载请附上原文链接哦~
Tisfy:https://letmefly.blog.csdn.net/article/details/129972641

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Tisfy

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值