python list拷贝赋值问题

本文探讨了Python中list拷贝和赋值的问题,指出使用等号或append方法会导致引用复制,而非真正复制。通过示例展示了如何通过copy和deepcopy方法正确创建list副本,以避免意外修改。总结了不同赋值方式下修改list及其子列表的影响,强调了深拷贝在处理二维list时的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在python中,用等号或者append方法进行list的赋值,实际上是将原list的引用赋给了新的变量,在后续使用append方法改变这两个list中的任何一个时,另一个也会同时改变:

b = [0,1,2]
a = b
b.append(3)
print(a)
print(b)
>>a = [0,1,2,3]
>>b = [0,1,2,3]
a.append(4print(a)
print(b)
>>a = [0,1,2,3,4]
>>b = [0,1,2,3,4]
a[0]=5
>>a = [5,1,2,3,4]
>>b = [5,1,2,3,4]

用append方法把b添加进a也会发生同样的情况:

a = []
b = [0,1,2]
a.append(b)
b.append(3)
print(a)
print(b)
>>a = [[0,1,2,3]]
>>b = [0,1,2,3]
a[0].append(4print(a)
print(b)
>>a = [[0,1,2,3,4]]
>>b = [0,1,2,3,4]

避免这种情况的方法是用另一个变量把b给copy出来:

b = [0,1,2]
c = b.copy()
a = c
b.append(3)
print(a)
print(b)
print(c)
>>a = [0,1,2]
>>b = [0,1,2,3]
>>c = [0,1,2]
a.append(4)
print(a)
print(b)
print(c)
>>a = [0,1,2,4]
>>b = [0,1,2,3]
>>c = [0,1,2,4]

把a=c这行换成a.append(c)情况类似,但是这好像是append这类直接修改列表的方法才会出现的(除append之外还有pop),如果重新给b赋其他值,则不会出现上述问题:(应该是重新赋值时新建了一块地址叫b,切断了a和之前的变量b的关系)

b = [0,1,2]
a = b
b = b[1:]
print(a)
print(b)
>>a = [0,1,2]
>>b = [1,2]
b = [3,4,5]
print(a)
print(b)
>>a = [0,1,2]
>>b = [3,4,5]

如果b是个二维list呢?

b = [[0,1],[2,3]]
a = b.copy()
id(a)==id(b)
>>False
id(a[0])==id(b[0])
>>True

也就是说使用copy的时候,只是把a和b的地址分开了,他们里面实际的子列表还是引用的关系,修改其中一个子列表必然会影响到另一个:

a[0].append(2)
print(a)
print(b)
>>a = [[0,1,2],[2,3]]
>>b = [[0,1,2],[2,3]]

但是修改a却不会同时修改b:

a.append(4)
>>a = [[0,1,2],[2,3],4]
>>b = [[0,1,2],[2,3]]

所以遇到二维list的时候乖乖用deepcopy()吧。
总结一下:
1.用等号(或appne)将list赋值:a=b(a.append(b)),a和b互为引用关系,id(a)==id(b), id(a[0])==id(b[0])。
2.用copy将list赋值:a=b.copy(),则id(a)!=id(b), id(a[0])==id(b[0])。
3.当id(a)==id(b)时,直接修改a如a.append(x)这类方法,会同时修改b,改变a中的值如a[0]=x,也会同时修改b。
4.当id(a)!=id(b), id(a[0])==id(b[0])时,直接修改a如a.append(x)这类方法,或者修改a[0],都不会同时修改b,但操作到a[0]中的元素,改变a[0]中的值如a[0][0]=x,会同时修改b。
5.也就是说,当两个变量的某一层次是引用关系时,使用append、pop修改当前层级会同时修改另一变量,使用直接赋值修改当前层级不会同时修改另一变量。当两个变量的某一层次是引用关系时(id(a)==id(b)),使用赋值修改下一层级的值(a[0]=x),会同时修改另一变量。
6.用深拷贝a=b.deepcopy()肯定没错。
感觉list真是个神奇的东西。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值