首页 > 编程 > Python > 正文

python函数传参是传值还是传引用?

2019-11-09 20:24:53
字体:
来源:转载
供稿:网友

python不允许程序员选择采用传值还是传引用。Python参数传递采用的肯定是“传对象引用”的方式:如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值--相当于通过“传引用”来传递对象。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象--相当于通过“传值'来传递对象。

值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值。

引用传递(pass-by-reference)过程中,被调函数的形式参数虽然也作为局部变量在堆栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过堆栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。

先看一个简单的例子:

   

复制代码
from ctypes import *import os.path  import sysdef test(c):    PRint "test before "    print id(c)    c+=2    print "test after +"    print id(c)    return cdef printIt(t):    for i in range(len(t)):        print t[i]if __name__=="__main__":    a=2    print "main before invoke test"    print id(a)    n=test(a)    print "main afterf invoke test"    print a    print id(a)    复制代码

运行后结果如下:

 

复制代码
>>> main before invoke test39601564test before 39601564test after +39601540main afterf invoke test239601564复制代码

 

 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)40650152>>> id(b)40650152>>> a=2>>> id(a)40650140

复制代码

       a和b都是int类型的值,值都是1,而且内存地址都是一样的,这已经表明了在python中,可以有多个引用指向同一个内存(画了一个很挫的图,见谅),在给a赋值为2后,再次查看a的内存地址,都已经变化了

      

       而基于最前面的例子,大概可以这样描述:

        

      那python函数传参就是传引用?然后传参的值在被调函数内被修改也不影响主调函数的实参变量的值?再来看个例子。

复制代码
from ctypes import *import os.path  import sysdef test(list2):    print "test before "    print id(list2)    list2[1]=30    print "test after +"    print id(list2)    return list2def printIt(t):    for i in range(len(t)):        print t[i]if __name__=="__main__":    list1=["loleina",25,'female']    print "main before invoke test"    print id(list1)    list3=test(list1)    print "main afterf invoke test"    print list1    print id(list1)    复制代码

      实际值为:

复制代码
>>> main before invoke test64129944test before 64129944test after +64129944main afterf invoke test['loleina', 30, 'female']64129944复制代码

      发现一样的传值,而第二个变量居然变化,为啥呢?

      实际上是因为python中的序列:列表是一个可变的对象,就基于list1=[1,2] list1[0]=[0]这样前后的查看list1的内存地址,是一样的。

   

复制代码
>>> list1=[1,2]>>> id(list1)64185208>>> list1[0]=[0]>>> list1[[0], 2]>>> id(list1)64185208结论:python不允许程序员选择采用传值还是传引用。Python参数传递采用的肯定是“传对象引用”的方式。这种方式相当于传值和传引用的一种综合。如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值--相当于通过“传引用”来传递对象。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象--相当于通过“传值'来传递对象。
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表