1.浅拷贝
- 分析浅拷贝时,需要分成两种情况来讨论:
- 情况1:如果最外层的数据类型是可变对象(列表、字典、集合等),使用浅拷贝时会开辟新的地址去存放。
- 情况2:如果最外层的数据类型是不可变对象(字符串、数字、元组等),使用浅拷贝时不会开辟新的地址空间。
- 情况1的实例如下:
import copy a = [1, 2] b = [3, 4] c = [a, b] # 注意这里的c就是一个可变对象! d = copy.copy(c) print("变量c:", c) print("变量c在内存中的地址:",id(c)) print("变量a在内存中的地址:",id(a)) print("变量b在内存中的地址:",id(b)) print("----------------------------------") print("变量d:", d) print("变量d在内存中的地址:",id(d)) print("变量d[0]在内存中的地址:",id(d[0])) print("变量d[1]在内存中的地址:",id(d[1]))
- 结果:
变量c: [[1, 2], [3, 4]] 变量c在内存中的地址: 43949256 变量a在内存中的地址: 43990920 变量b在内存中的地址: 43949192 ---------------------------------- 变量d: [[1, 2], [3, 4]] 变量d在内存中的地址: 43988488 变量d[0]在内存中的地址: 43990920 变量d[1]在内存中的地址: 43949192
- 原因分析:需要拷贝的对象c是一个list,即可变对象。因为c的最外层是[a,b]。因此在执行浅拷贝时,仅仅把最外层拷贝过来,而里面的值没有拷贝过来。
- 情况2的实例如下:
import copy a = [1, 2] b = [3, 4] c = (a, b) # 注意这里的c就是一个不可变对象! d = copy.copy(c) print("变量c:", c) print("变量c在内存中的地址:",id(c)) print("变量a在内存中的地址:",id(a)) print("变量b在内存中的地址:",id(b)) print("----------------------------------") print("变量d:", d) print("变量d在内存中的地址:",id(d)) print("变量d[0]在内存中的地址:",id(d[0])) print("变量d[1]在内存中的地址:",id(d[1]))
- 结果:
变量c: ([1, 2], [3, 4]) 变量c在内存中的地址: 42186696 变量a在内存中的地址: 42286984 变量b在内存中的地址: 42245256 ---------------------------------- 变量d: ([1, 2], [3, 4]) 变量d在内存中的地址: 42186696 变量d[0]在内存中的地址: 42286984 变量d[1]在内存中的地址: 42245256
- 原因分析:因为c = (a,b),c是一个元组tuple,即c的最外层是一个不可变对象。当d在进行浅拷贝操作时,直接引用c的地址,不会再为d开辟新的地址空间。
2.深拷贝
- 分析深拷贝时,需要分成:三种情况来讨论:
- 情况1:如果最外层的数据类型是可变对象(列表、字典、集合等),使用深拷贝时,内部和外部的数据都会被拷贝过来。
- 情况2:如果最外层的数据类型是不可变对象(字符串、数字、元组等),但内部的数据类型是可变对象时。使用深拷贝时,会开辟新的地址空间。
- 情况3:如果最外层的数据类型是不可变对象(字符串、数字、元组等),内部的数据类型也是不可变对象时,和浅拷贝的第二种情况一样,不会开辟新的地址空间,是对原始数据对象的引用。
- 情况1实例如下:
import copy a = [1, 2] # a是可变对象 b = copy.deepcopy(a) print("变量a:", a) print("变量a在内存中的地址:",id(a)) print("----------------------------------") print("变量b:", b) print("变量b在内存中的地址:",id(b))
- 结果:
变量a: [1, 2] 变量a在内存中的地址: 43925384 ---------------------------------- 变量b: [1, 2] 变量b在内存中的地址: 43883656
- 原因分析:由于最外层是可变对象list,因此对a进行深拷贝会开辟一个新的地址空间,修改a的值,不会影响b
- 情况2实例如下:
import copy a = [1, 2] b = [3, 4] c = (a, b) # 注意这里外层的c就是一个不可变对象!但是里面的a和b都是可变对象! d = copy.deepcopy(c) print("变量c:", c) print("变量c在内存中的地址:",id(c)) print("变量a在内存中的地址:",id(a)) print("变量b在内存中的地址:",id(b)) print("----------------------------------") print("变量d:", d) print("变量d在内存中的地址:",id(d)) print("变量d[0]在内存中的地址:",id(d[0])) print("变量d[1]在内存中的地址:",id(d[1]))
- 结果:
变量c: ([1, 2], [3, 4]) 变量c在内存中的地址: 40729736 变量a在内存中的地址: 40845192 变量b在内存中的地址: 40803464 ---------------------------------- 变量d: ([1, 2], [3, 4]) 变量d在内存中的地址: 40803080 变量d[0]在内存中的地址: 40845256 变量d[1]在内存中的地址: 40842760
- 原因分析:因为最外层的c是一个元组,不可变对象。但是里面的数据a和b是两个list可变对象。因此在深拷贝时,是会开辟新的地址空间的。
- 情况3实例如下:
import copy a = (1, 2) b = (3, 4) c = (a, b) # 注意这里外层的c就是一个不可变对象!但是里面的a和b也都是不可变对象! d = copy.deepcopy(c) print("变量c:", c) print("变量c在内存中的地址:",id(c)) print("变量a在内存中的地址:",id(a)) print("变量b在内存中的地址:",id(b)) print("----------------------------------") print("变量d:", d) print("变量d在内存中的地址:",id(d)) print("变量d[0]在内存中的地址:",id(d[0])) print("变量d[1]在内存中的地址:",id(d[1]))
- 结果:
变量c: ((1, 2), (3, 4)) 变量c在内存中的地址: 41129928 变量a在内存中的地址: 41072904 变量b在内存中的地址: 41130632 ---------------------------------- 变量d: ((1, 2), (3, 4)) 变量d在内存中的地址: 41129928 变量d[0]在内存中的地址: 41072904 变量d[1]在内存中的地址: 41130632
- 原因分析:因为c = (a,b),c是一个元组tuple,即c的最外层是一个不可变对象;同时,里面的a和b也是元组对象,不可变对象。当d在进行深拷贝操作时,直接引用c的地址,不会再为d开辟新的地址空间,与浅拷贝的第二种情况一样。