python深浅拷贝

python深浅拷贝

 

对象的内存使用方式

python一切皆为对象,python中的变量就相当于一个引用,指向一个对象。

 

a = 1

整数1为一个对象。而a是一个引用,利用赋值语句,引用a指向对象1

 

id函数返回对象的内存地址

a = 1
b = 1
print(id(a))
print(id(b))

上面程序返回

1520528400

1520528400

可见a和b实际上是指向同一个对象的两个引用。

 

a = 1
b = a

在python中,对象赋值实际上是对象的引用。当创建一个对象,然后把它赋给另一个变量的时候,python并没有拷贝这个对象,而只是拷贝了这个对象的引用

 

对象的引用计数

在Python中,每个对象都有存有指向该对象的引用总数,即引用计数(reference count)。

我们可以使用sys包中的getrefcount(),来查看某个对象的引用计数。需要注意的是,当使用某个引用作为参数,传递给getrefcount()时,参数实际上创建了一个临时的引用。因此,getrefcount()所得到的结果,会比期望的多1。

from sys import getrefcount
a = [1, 2, 3]
print(getrefcount(a))
b = a
print(getrefcount(b))

由于上述原因,两个getrefcount将返回2和3,而不是期望的1和2。

 

垃圾回收

从基本原理上,当Python的某个对象的引用计数降为0时,说明没有任何引用指向该对象,该对象就成为要被回收的垃圾了。比如某个新建对象,它被分配给某个引用,对象的引用计数变为1。如果引用被删除,对象的引用计数为0,那么该对象就可以被垃圾回收。比如下面的表

a = [1, 2, 3]
del a

del a后,已经没有任何引用指向之前建立的[1, 2, 3]这个表。用户不可能通过任何方式接触或者动用这个对象。这个对象如果继续待在内存里,就成了不健康的脂肪。当垃圾回收启动时,Python扫描到这个引用计数为0的对象,就将它所占据的内存清空。

 

深浅拷贝理论

复制:拷贝对象的引用的值

浅拷贝: 拷贝对象

深拷贝: 拷贝对象和其子对象

 

深浅拷贝案例

a = [[1, 2], 3, 4]

 

b=a,复制只拷贝a的值,b增删改元素影响a,b[0]增删改元素会影响a[0]

c=a.copy(),c=a[: ],c=copy.copy(a),浅拷贝,c增删改不影响a,c[0]增删改元素影响a[0]

d=copy.deepcopy(a),深拷贝,c增删改元素,c[0]增删改元素,都不影响a

 

深浅拷贝代码验证

import copy
a = [[1, 2], 3, 4]
b = a
c = a.copy()
d = copy.deepcopy(a)

b.append(5)
b[0].append(6)
print(b)  # [[1, 2, 6], 3, 4, 5]
print(a)  # [[1, 2, 6], 3, 4, 5]

c.append(7)
c[0].append(8)
print(c)  # [[1, 2, 6, 8], 3, 4, 7]
print(a)  # [[1, 2, 6, 8], 3, 4, 5]

d.append(9)
d[0].append(10)
print(d)  # [[1, 2, 10], 3, 4, 9]
print(a)  # [[1, 2, 6, 8], 3, 4, 5]

 

 

### Python中的深拷贝与浅拷贝 Python 提供了两种复制对象的方式:浅拷贝(shallow copy)和深拷贝(deep copy)。这两种方式在处理复杂数据结构时表现不同,因此理解两者的区别至关重要。 #### 一、浅拷贝 (Shallow Copy) 浅拷贝是指创建一个新的复合对象,然后将原始对象内部的引用插入到新对象中。这意味着如果原对象包含其他容器类型的子对象,则这些子对象不会被复制,而是共享相同的内存地址[^1]。 以下是一个简单的例子展示了列表的浅拷贝行为: ```python import copy original_list = [1, 2, [3, 4]] shallow_copied_list = copy.copy(original_list) original_list[2][0] = 'Modified' print("Original List:", original_list) # Output: Original List: [1, 2, ['Modified', 4]] print("Shallow Copied List:", shallow_copied_list) # Output: Shallow Copied List: [1, 2, ['Modified', 4]] ``` 可以看到,在修改 `original_list` 的嵌套列表之后,`shallow_copied_list` 中对应的嵌套列表也被改变了,因为两者指向的是同一个内存位置。 #### 二、深拷贝 (Deep Copy) 相比之下,深拷贝不仅会创建新的外部容器,还会递归地复制所有的子对象。这样即使修改了原来的对象也不会影响到它的副本[^2]。 下面是同样的场景下使用深拷贝的结果对比: ```python import copy original_list = [1, 2, [3, 4]] deep_copied_list = copy.deepcopy(original_list) original_list[2][0] = 'Modified' print("Original List:", original_list) # Output: Original List: [1, 2, ['Modified', 4]] print("Deep Copied List:", deep_copied_list) # Output: Deep Copied List: [1, 2, [3, 4]] ``` 在这里可以看出,尽管我们更改了 `original_list` 内部的一个元素,但是由于采用了深拷贝策略,所以 `deep_copied_list` 并未受到影响[^2]。 #### 总结 - **浅拷贝**只复制顶层的对象,并保留对其余部分(尤其是可变类型如列表或字典等)的引用关系。 - **深拷贝**则完全独立地重建整个层次结构树形图谱下的每一个节点实例化版本。 选择哪种方式进行操作取决于具体的应用需求以及目标数据集本身的特性等因素考虑决定最佳实践方案。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值