最近使用python写代码,突然遇到形参传递给函数后,在函数内部修改完改形参,发现实参变量也随着改变的问题。后来一查,才了解到可变对象和不可变对象的区分,以及通过copy办法解决这个问题。
python参数传递
python中,如果把数字、字符串传入到函数中。在函数内部改变其值,当函数运行结束时,该变量不会发生改变。
In [1]: def sum(a,b):
...: print(a+b)
...:
In [2]: a=2
In [3]: b=3
In [4]: sum(a,b)
5
In [5]: print(a,b)
2 3
这些变量在函数调用后不会改变,这样的变量称为 不可变对象(数字、字符、字符串、元组)
但是如果传递给函数的变量是可变对象(列表、字典),函数内修改形参变量,是会将这个变量本身进行修改。例如:
In [17]: def add(a):
...: a[0]=a[0]+1
...:
In [18]: a=[1,2,3]
In [19]: add(a)
In [20]: a
Out[20]: [2, 2, 3]
总结:
python中一切都是对象,函数中传递的仅仅是对象的引用,当形参通过传递来的引用修改了对象的内容,实参会跟着改变,因为此时形参和实参指向的是同一个对象。当形参指向不同对象的时候,实参不会被改变。
类似于形参变量和实参变量引用的是同一个地址,所以在函数内部改变了可变对象,实参变量本身也发生改变。
解决办法:深拷贝/浅拷贝
copy.copy()浅拷贝
浅拷贝创建了一个跟原对象一样的类型,但是其内容是对原对象元素的引用。拷贝的对象本身是新的,但内容不是。
In [21]: import copy
In [22]: b = copy.copy(a)
In [23]: add(b)
In [24]: b
Out[24]: [3, 2, 3]
In [25]: a
Out[25]: [2, 2, 3]
深拷贝,将内存中所有的数据重新创建一份(排除最后一层,即;python内部对字符串和数字的优化),如下图:
In [26]: c=copy.deepcopy(a)
In [27]: add(c)
In [28]: c
Out[28]: [3, 2, 3]
In [29]: a
Out[29]: [2, 2, 3]
注意:赋值只是创建一个变量,但是该变量还是指向原来的内存地址。所以通过赋值办法不能解决这个问题。