先看个例子:
import copy #导入copy模块
a = [1, [1,2], 3 ] #原始对象
b = a #赋值,传对象的引用
c = copy.copy(a) #对象拷贝,浅拷贝
d = copy.deepcopy(a) #对象拷贝,深拷贝
a.append(4) #修改原始对象a,在尾部添加一个元素
a[1].append(3) #修改对象a中的列表对象
print(a) #[1, [1,2,3], 3, 4] 原始列表修改后的状态
print(b) #[1, [1,2,3], 3, 4] 与原始列表保持状态一致
print(c) #[1, [1,2,3], 3] 内嵌的列表对象被改变,但尾部没有增加元素4
print(d) #[1, [1,2], 3] 内容保持在最初拷贝的状态,不受原始对象修改的影响
引用
Python中等号“=”的实际作用是传递对象的引用,而不是将目标对象的所有内容复制一遍再传给新对象。
执行语句“b = a”的时候,相当于a告诉b对象在哪,然后a和b都指向了内存中同一个对象。因此,通过a修改对象的时候,b指向的对象肯定会百分百受影响,毕竟指向的本来就是同一个对象嘛。
浅拷贝
我们一般使用copy模块的copy方法来实现浅拷贝。顾名思义,浅拷贝就是浅层的拷贝,我们一般理解为顶层拷贝。回到上面的例子,a指向的对象是一个列表,而列表中有个列表元素[1,2],因此a[1]存放的实际上也是个引用,这个引用指向了内嵌的列表对象[1,2]。
执行语句“c = copy.copy(a)”的时候,c了拷贝a的顶层元素(1、指向[1,2]的引用、3),但是a修改[1,2]的话,c一样可以“看见”。
补充:系列的切片操作,如c = a[:]也是浅拷贝。
深拷贝
我们一般使用copy模块的deepcopy方法来实现深拷贝。深拷贝就是完全拷贝,深拷贝才是拷贝包含嵌套的对象的正确方式。
执行语句“d = copy.deepcopy(a)”的时候,a的所有元素都将被拷贝到另一个地方,然后d指向这个地方。所以无论原始对象有什么修改,对d引用的对象都不会有任何影响。