上面的图中,我们将箭头连线看作是引用。
如果你只是简单是 b = a
,实际上两者的引用是一样的,相当于 b 只是 a 的另外一个名字,不管是对 a 或者 b 内的可变元素还是不可变元素修改,打印 a, b 两者都是一样的。
但是如果你使用浅拷贝,也就是 copy.copy()
来进行浅拷贝,实际上 a, b 是两个对象了,他们两个的 id 是不一样的。但是因为对象是列表,内部还有子对象,在浅拷贝的时候,会对他们的引用进行复制,所以子对象的引用是没有变的,但是你修改不可变元素的时候,实际上不是在修改它的值,而是将它的引用去掉,然后用新的引用指向新的值。如果修改的是可变对象,则可以直接在上面进行修改。所以 a[0] 修改之后,c[0] 还会是原来的值。但是如果修改 a[1], c[1] 也会发生改变。
如果将拷贝理解为分身的话,浅拷贝只会进行一层的分身,内部嵌套的元素是直接复制的,相当于赋值。
但是如果是深拷贝的,相当于递归进行分身,不管是顶层对象,还是子对象,都会递归进行拷贝,也就是所谓的“深”。
深拷贝之后,两个对象的子对象的引用都完全不一样了,两者完全没有任何关系了。