在理解python拷贝和赋值(=)之前,可以先理解一下python的变量和赋值:
在执行a = 'ABC'的时候,python会创建'ABC'对象和a变量,并让a变量指向‘ABC’对象,此时可以把a理解成‘ABC’的一个引用,当引用为0时,‘ABC’可能会被python回收内存。
在执行b=a的时候,b指向a指向的对象,也就是‘ABC’。此时a和b无关了,只有你改变‘ABC’的时候a和b才会同时改变,但是我们知道‘ABC’是一个不可变对象,所以接下来讨论a和b指向可变对象才是重点。
在讨论之前,还想申明一个python在设计变量和对象时的理念(自己猜的不一定准确):
不可变对象不会被复制。
比如:
a = 'ABC'
b = 'ABC'
id(a) == id(b)
会返回True
这个理念贯穿python深浅拷贝的设计思路。
接着上面可变对象(列表,字典,类等)继续讨论:
可变对象包含不可变对象
我们拿list来举例,如果list里是不可变对象,那么在浅拷贝和深拷贝一样,不会有区别:
a = [1,2,3]
b = copy.copy(a)
c = copy.deepcopy(a)
id(a[0]) == id(b[0]) # True
id(a[0]) == id(c[0]) # True
id(a) == id(b) # False
id(a) == id(c) # False
我画了一张图来理解:
![b9e27d99c3d78473c51dc26a71cb8f74.png](https://i-blog.csdnimg.cn/blog_migrate/a176106712a6930a6f53f66fe2276b2c.png)
当可变对象里是不可变对象时,浅拷贝和深拷贝的机制相同:都是先创建一个新的list,里面的引用都指向原来的不可变对象,符合不可变对象不会复制的理念。
可变对象包含可变对象
list里包含list的情况,才真正谈及到了浅拷贝和深拷贝:
a = [[1,2,3], [3,4,5]]
b = copy.copy(a)
c = copy.deepcopy(a)
id(a[0]) == id(b[0]) # True
id(a[0]) == id(c[0]) # False
浅拷贝时,只拷贝子对象的引用,深拷贝会拷贝所有的对象。同样画了个图进行理解:
![49a48eb2e568f54fd00b68793ba0ef90.png](https://i-blog.csdnimg.cn/blog_migrate/4196efe31d97fe852d2555614f72a6a1.png)
这也符合其他语言浅拷贝和深拷贝的概念。正是因为不可改变对象不复制的理念,让python在拷贝对象时考虑到对象是否有必要进行深拷贝,同时也让python中浅拷贝和深拷贝变得难以理解。
最后留一题思考题,当可变变量中同时包含可变变量和不可变变量时,会怎么样:
a = [1, [1,2,3]]
b = copy.copy(a)
c = copy.deepcopy(a)
id(a[0]) == id(b[0])
id(a[1]) == id(b[1])
id(a[0]) == id(c[0])
id(a[1]) == id(c[1])