在Python中,无论是对象赋值,作为为参数传递,作为函数返回值,都是引用传递的.如果需要拷贝对象,需要使用标准库中的copy模块(import copy)。
1.浅拷贝 :
使用copy.copy,它可以进行对象的浅拷贝(shallow copy),它复制了对象,但对于对象中的元素,依然使用引用(换句话说修改拷贝对象元素,则被拷贝对象元素也被修改)
2.深拷贝 :
使用copy.deepcopy,它可以进行深拷贝,不仅拷贝了对象,同时也拷贝了对象中的元素,获得了全新的对象,与被拷贝对象完全独立,但这需要牺牲一定的时间和空间。
3.特殊拷贝:
如要复制列表L,使用list(L),要复制一个字典d,使用dict(d),要复制一个集合s,使用set(s)。
总结一下的话:如果你要复制某个对象object, 它属于python内建的类型type,那么你可以使用type(object)来 获得一个拷贝。
4 举例:
import copy
#此事件为浅拷贝
a = [1,2,3,4,['a','b']] #创建一个对象a
print("原始数据:",a)
b = copy.copy(a)#浅拷贝
print(id(a)==id(b))#False,内存地址:False,说明,b在内存中已经独立出一部分复制了a的数据
b.append(5)#追加5
print("拷贝后的数据修改第一层:",b)
print("拷贝后的原始数据:",a)
"""
拷贝后的b追加5,a中不会改变,说明b已经复制了a的第一层,
那第二层呢?接着往下看...
"""
b[4][1] = "c"#再修改子列表中的b元素
print("拷贝后的数据修改第二层:",b)
print("拷贝后的原始数据:",a)
"""
b中子列表元素b改为了c
a中子列表b也改成了c,说明没有被拷贝,只是引用了对象中的元素,所以说这块是共享的
"""
原始数据: [1, 2, 3, 4, [‘a’, ‘b’]]
False
拷贝后的数据修改第一层: [1, 2, 3, 4, [‘a’, ‘b’], 5]
拷贝后的原始数据: [1, 2, 3, 4, [‘a’, ‘b’]]
拷贝后的数据修改第二层: [1, 2, 3, 4, [‘a’, ‘c’], 5]
拷贝后的原始数据: [1, 2, 3, 4, [‘a’, ‘c’]]
import copy
#此事件为深拷贝
a = [1,2,3,4,['a','b']] #创建一个对象a
print("原始数据:",a)
c = copy.deepcopy(a)#深拷贝
c.append(5)#追加5
print(id(a)==id(c))#False
print("拷贝后的数据修改第一层:",c)
print("拷贝后的原始数据:",a)
"""拷贝后的c追加5,a没有被修改
"""
c[4][1] = "c"#再修改子列表中的b元素
print("拷贝后的数据修改第二层:",c)
print("拷贝后的原始数据:",a)
"""
c中子列表元素b改为了c
a没有被修改
"""
原始数据: [1, 2, 3, 4, [‘a’, ‘b’]]
False
拷贝后的数据修改第一层: [1, 2, 3, 4, [‘a’, ‘b’], 5]
拷贝后的原始数据: [1, 2, 3, 4, [‘a’, ‘b’]]
拷贝后的数据修改第二层: [1, 2, 3, 4, [‘a’, ‘c’], 5]
拷贝后的原始数据: [1, 2, 3, 4, [‘a’, ‘b’]]
由此可见深拷贝就是数据完完全全独立拷贝出来一份。不会由原先数据变动而变动