一、提个需求先~
有一个列表 a = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19],现在想要修改其中的 1 为 999。于是乎。我是这样写的:
a = list(range(20))
a[1] = 999
print(a)
代码报错:
原因是:999 是个 int 型数值,不具备迭代能力!!!
二、解决办法
与 int 型数据不同的是,str 型数据是具备迭代能力的。另外修改成 (999, ) 也可以~如下:
a = list(range(20))
a[1] = (999, )
print(a)
a[1] = '999'
print(a)
(参考:TypeError: can only assign an iterable)
(参考:Python_异常处理:TypeError: can only assign an iterable)
三、需求升级
将 a 中 1、2、3 都改成 0。尝试 (0, ) 这种方法。
a = list(range(20))
print(a)
b = a # 浅拷贝,会修改 a 的原始数据
b[1:4] = (0, )
print(a)
print(b)
发现问题及分析:
(1)b 长度变短;(这种方法相当于将 1/2/3 三个元素替换成了 1 个值,所以长度变短,所以这种方法只适用于事先知道修改元素长度的情况。)
(2)修改 b 的同时也修改了 a。(Python 中所有的直接赋值都是浅拷贝!)
四、更好的解决办法
1. 针对上述问题一中 b 长度变短这一问题,使用 list[:] = (0, 0, ...) 或 np.array() 解决。
a = list(range(20))
c = a
c[1:4] = (0, 0, 0) # 这种方法需要事先知道需要修改元素的个数,如果个数未知,那么不适用!
print(a)
print(c)
import numpy as np
a = np.array(list(range(20)))
print(a)
a[1:4] = 0
print(a)
2. 针对上述问题二中 a 一起被修改的问题,使用深拷贝解决。
import numpy as np
a = np.array(list(range(20)))
print(a)
b = a.copy() # 深拷贝,不修改 a 的原始数据
b[1:4] = 0
print(a)
print(b)
(参考:python按引用赋值和深、浅拷贝)