尽管使用 .append 需要一个方法调用,但它实际上比使用扩展赋值运算符 += 更有效。
但是使用.append 有一个更好的理由:当您附加到的列表不在本地范围内时,您可以这样做,因为它只是在外部范围内调用对象的方法,而除非您将它们声明为全局(或非局部),否则您不能对不在本地范围内的对象执行赋值,这种做法通常最好避免。
这是一个例子:
mat = []
def test_append():
for i in range(10):
#mat += [i]
mat.append(i)
def test_iadd():
for i in range(10):
mat += [i]
test_append()
print(mat)
test_iadd()
print(mat)
输出
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
Traceback (most recent call last):
File "./qtest.py", line 29, in <module>
test_iadd()
File "./qtest.py", line 25, in test_iadd
mat += [i]
UnboundLocalError: local variable 'mat' referenced before assignment
当然,我们可以将mat 作为参数传递给函数:
mat = []
def test_append():
for i in range(10):
#mat += [i]
mat.append(i)
def test_iadd2(mat):
for i in range(10):
mat += [i]
test_append()
print(mat)
test_iadd2(mat)
print(mat)
输出
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
test_iadd2 较慢的一个原因是它必须在 every 循环上构造一个新的[i] 列表。
FWIW,这是字节码:
test_append
21 0 SETUP_LOOP 33 (to 36)
3 LOAD_GLOBAL 0 (range)
6 LOAD_CONST 1 (10)
9 CALL_FUNCTION 1
12 GET_ITER
>> 13 FOR_ITER 19 (to 35)
16 STORE_FAST 0 (i)
23 19 LOAD_GLOBAL 1 (mat)
22 LOAD_ATTR 2 (append)
25 LOAD_FAST 0 (i)
28 CALL_FUNCTION 1
31 POP_TOP
32 JUMP_ABSOLUTE 13
>> 35 POP_BLOCK
>> 36 LOAD_CONST 0 (None)
39 RETURN_VALUE
test_iadd2
26 0 SETUP_LOOP 33 (to 36)
3 LOAD_GLOBAL 0 (range)
6 LOAD_CONST 1 (10)
9 CALL_FUNCTION 1
12 GET_ITER
>> 13 FOR_ITER 19 (to 35)
16 STORE_FAST 1 (i)
27 19 LOAD_FAST 0 (mat)
22 LOAD_FAST 1 (i)
25 BUILD_LIST 1
28 INPLACE_ADD
29 STORE_FAST 0 (mat)
32 JUMP_ABSOLUTE 13
>> 35 POP_BLOCK
>> 36 LOAD_CONST 0 (None)
39 RETURN_VALUE
上述字节码转储是使用dis 模块生成的:
from dis import dis
mat = []
def test_append():
for i in range(10):
#mat += [i]
mat.append(i)
def test_iadd2(mat):
for i in range(10):
mat += [i]
print('test_append')
dis(test_append)
print('\ntest_iadd2')
dis(test_iadd2)