Python中要区分两个概念,变量和对象,对象是在内存中开辟的一块区域,变量指向这块区域,当变量间直接赋值时,等同于两个变量指向同一个对象即同一块区域。变量可以被删除,对象只能通过垃圾回收机制回收。
在C和C++等语言中,参数传递通常分为值传递和引用传递,值传递是直接赋值,被传递的参数修改不影响原变量,引用传递是传递变量指针,被传递的参数修改影响原变量。
但是在Python中不做区分,只有一种方式,就是通过直接赋值来传递参数,直接赋值之后两个变量指向同一个对象,如果修改两个变量中的任一个导致了对象改变,则另一个指向该对象的变量也会跟着改变,反之,则不变。
**一般只有像列表、集合、字典等本身可变的变量会导致对象改变以外,其他不可变(整型、字符串、元组等)的变量一般都不会导致对象的改变,因为通过某些操作(+= 等等)更新不可变对象的值时,会返回一个新的对象,对于原对象没影响,指向原对象的变量自然也不会改变。**如:
def my_func1(b):
b = 2
a = 1
my_func1(a)
# 输出
a
1
def my_func3(l2):
l2.append(4)
l1 = [1, 2, 3]
my_func3(l1)
# 输出
l1
[1, 2, 3, 4]
这里l2指向一个可变的列表,l1和l2同时指向这个列表,l2改变会导致这个列表对象改变,因此l1也会跟着改变。
def my_func4(l2):
l2 = l2 + [4]
l1 = [1, 2, 3]
my_func4(l1)
l1
[1, 2, 3]
但是,这里 l2 = l2 + [4],表示创建了一个“末尾加入元素 4“的新列表,并让 l2 指向这个新的对象,l1 指向的原来的对象不变,因此l1不变。
def my_func5(l2):
l2 = l2 + [4]
return l2
l1 = [1, 2, 3]
l1 = my_func5(l1)
l1
[1, 2, 3, 4]
因此,一般当一个变量指向一个可变变量时,要采用上述这种书写方式,如果需要改变l1,让上述函数返回一个新列表,再赋予 l1 即可,防止出错。