python是“传对象引用”
首先还是应该科普下函数参数传递机制,传值和传引用是什么意思?
函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题。基本的参数传递机制有两种:值传递和引用传递。
值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值。
引用传递(pass-by-reference)过程中,被调函数的形式参数虽然也作为局部变量在堆栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过堆栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。
在python中实际又是怎么样的呢?
from ctypes import *
import os.path
import sys
def test(c):
print("test before")
print(id(c))
c+=2
print("test after")
print(id(c))
return c
if __name__ == '__main__':
a = 2
print("main before invoke test")
print(id(a))
n = test(a)
print("main after invoke test")
print(a)
print(id(a))
结果如下:
main before invoke test
1483651568
test before
1483651568
test after
1483651632
main after invoke test
2
1483651568
id函数可以获得对象的内存地址.很明显从上面例子可以看出,将a变量作为参数传递给了test函数,传递了a的一个引用,把a的地址传递过去了,所以在函数内获取的变量C的地址跟变量a的地址是一样的,但是在函数内,对C进行赋值运算,C的值从2变成了4,实际上2和4所占的内存空间都还是存在的,赋值运算后,C指向4所在的内存。而a仍然指向2所在的内存,所以后面打印a,其值还是2.
如果还不能理解,先看下面例子:
>>> a=1
>>> b=1
>>> id(a)
>>> id(b)
>>> a=2
>>> id(a)
那python函数传参就是传引用?然后传参的值在被调函数内被修改也不影响主调函数的实参变量的值?再来看个例子。
def test1(list2):
print("test before")
print(id(list2))
list2[1] = 30
print("test after")
print(id(list2))
return list2
if __name__ == '__main__':
list1 = ["man",25,"woman"]
print("main before invoke test")
print(id(list1))
list3 =test1(list1)
print("main after invoke test")
print(list1)
print(id(list1))
结果如下:
main before invoke test
2093559227528
test before
2093559227528
test after
2093559227528
main after invoke test
['man', 30, 'woman']
2093559227528
发现一样的传值,而第二个变量居然变化,为啥呢?
实际上是因为python中的序列:列表是一个可变的对象,就基于list1=[1,2] list1[0]=[0]这样前后的查看list1的内存地址,是一样的。
>>> list1=[1,2]
>>> id(list1)
>>> list1[0]=[0]
>>> list1
[[0], 2]
>>> id(list1)
结论:python不允许程序员选择采用传值还是传引用。Python参数传递采用的肯定是“传对象引用”的方式。这种方式相当于传值和传引用的一种综合。如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值--相当于通过“传引用”来传递对象。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象--相当于通过“传值’来传递对象。