python函数参数传递特点_Python的函数参数传递

首先还是应该科普下函数参数传递机制,传值和传引用是什么意思?

函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题。基本的参数传递机制有两种:值传递和引用传递。

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

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

在python中实际又是怎么样的呢?

先看一个简单的例子

from ctypes import *

importos.pathimportsysdeftest(c):print "test before"

printid(c)

c+=2

print "test after +"

printid(c)returncdefprintIt(t):for i inrange(len(t)):printt[i]if __name__=="__main__":

a=2

print "main before invoke test"

printid(a)

n=test(a)print "main afterf invoke test"

printaprint id(a)

运行后结果如下:

>>>main before invoke test

test before

test after+main afterf invoke test39601564

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)

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

1320bbacea3d9c8251cef1f298e3779b.png

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

0c58bf949659afb25948ed3db1499172.png

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

from ctypes import *

importos.pathimportsysdeftest(list2):print "test before"

printid(list2)

list2[1]=30

print "test after +"

printid(list2)returnlist2defprintIt(t):for i inrange(len(t)):printt[i]if __name__=="__main__":

list1=["loleina",25,'female']print "main before invoke test"

printid(list1)

list3=test(list1)print "main afterf invoke test"

printlist1print id(list1)

实际值为:

>>>main before invoke test

test before

test after+main afterf invoke test

['loleina', 30, 'female']

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

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

>>> list1=[1,2]>>>id(list1)>>> list1[0]=[0]>>>list1

[[0],2]>>> id(list1)

字典也是可变对象:

>>> deffun2(num1,l1,d1):

... num1=123... l1[0]=123... d1['a']=123...print("inside:","num1=%f,l1=%s,d1=%s"%(num1,l1,d1))

...>>> num=111

>>> l=[1,1,1]>>> d={'a':111,'b':0}>>> print("before:","num=%f,l=%s,d=%s"%(num,l,d))

before: num=111.000000,l=[1, 1, 1],d={'a': 111, 'b': 0}>>>fun2(num,l,d)

inside: num1=123.000000,l1=[123, 1, 1],d1={'a': 123, 'b': 0}>>> print("after:","num=%f,l=%s,d=%s"%(num,l,d))

after: num=111.000000,l=[123, 1, 1],d={'a': 123, 'b': 0}

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值