列表的可变性
Python里的内置数据类型, 大致上可分为可变与不可变两种。
- 可变(mutable):列表、字典、集合
- 不可变(immutable):整数、浮点数、字符串、字节串、元组
列表是可变的,当我们初始化一个列表后,仍然可以调用.append()、.extend()等方法来修改它的内容。而字符串和整数等都是不可变的—我们没法修改一个已经存在的字符串对象。
学习Python时,理解类型的可变性是非常重要的一课。如果不掌握它,你在写代码时就会遇到很多与之相关的“Surprise“。
示例说明。
示例一:为字符串追加内容
在这个示例里, 我们定义一个往字符串追加内容的函数add_str(), 并在外层用一个字符串参数调用该函数:
def add_str(in_func_obj):
print(f'In add [before]: in_func_obj = "{in_func_obj}"')
in_func_obj += ' suffix'
print(f'In add [after]: in_func_obj = "{in_func_obj}"')
orig_obj = 'foo'
print(f'Outside [before]: orig_obj="{orig_obj}"')
add_str(orig_obj)
print(f'Outside [after]: orig_obj="{orig_obj}"')
运行上面的代码会输出这样的结果:
Outside [before]: orig_obj="foo"
In add [before]: in_func_obj = "foo"
In add [after]: in_func_obj = "foo suffix"
# 重要:这里的orig_obj变量还是原来的值
Outside [after]: orig_obj="foo"
这段代码里,原始字符串对象orig_obj被作为参数传给了add_str()函数的in_func_obj变量。随后函数内部通过+=操作修改了in_func_obj的值,为其增加了后缀字符串。但重点是:函数外的orig_obj变量所指向的值没有受到任何影响。
示例二:为列表追加内容
在这个例子中,我们保留一模一样的代码逻辑,但是把orig_obj换成了列表对象:
def add_list(in_func_obj):
print(f'In add [before]: in_func_obj = "{in_func_obj}"')
in_func_obj += ['baz']
print(f'In add [after]: in_func_obj = "{in_func_obj}"')
orig_obj = ['foo', 'bar']
print(f'Outside [before]: orig_obj="{orig_obj}"')
add_list(orig_obj)
print(f'Outside [after]: orig_obj="{orig_obj}"')
执行后会发现结果大不一样:
Outside [before]: orig_obj="['foo', 'bar']"
In add [before]: in_func_obj = "['foo', 'bar']"
In add [after]: in_func_obj = "['foo', 'bar', 'baz']"
# 注意:函数外的orig_obj变量的值已经被修改了
Outside [after]: orig_obj="['foo', 'bar', 'baz']"
当操作对象变成列表后,函数内的+=操作居然可以修改原始变量的值!