引用:
Python是不允许程序员选择采用传值还是传址的。Python参数传递采用的肯定是“传对象引用”的方式。实际上,这种方式相当于传值和传址的一种综合。如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值——相当于传址。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象——相当于传值。
所以python的传值和传址是根据传入参数的类型来选择的
传值的参数类型:数字,字符串,元组
传址的参数类型:列表,字典
做codecademy的练习时,声明了一个original变量,后来在函数f()里又声明了同名变量。执行结果中函数内original的值没有传递给外面的那个,我猜python分隔了函数内外的变量。
但孙咸鱼说原理没有这么简单,这是“传值还是传址”的问题。原来每个对象都是一个空间,值就放在空间中,空间有自己的地址。如果只传值,那么相当于copy了值出去;如果传址,就会让执行结果,顺着地址找到空间,修改对象本身。
p = [1]
def func1(tmp):
tmp.append(2)
func1(p)
for i in tmp:
print(i)
print(p)
执行结果为
1
2
[1, 2]
进行func1处理后,列表p的值被改变了。
如果对象是一个数字,就会这样:
tmp = 1
def func1():
print(tmp)
def func2(tmp):
tmp = 2
return tmp
func2(tmp)
func1()
返回结果为
1
还是原来的tmp值,因为变量tmp的类型是数字,进行func2(tmp)处理时,tmp只有值传入,没有地址,tmp变量本身不会变化。
此外,从例一中看,print()即便处于某个函数中,似乎也可以打印外面的变量值。但孙咸鱼做了下面的测试,发现并不总能这么做:
tmp = 1
def func1():
print tmp
def func3():
print tmp
tmp = 3
print tmp
func1()是ok的,func3()的写法不行。
因为func3()函数内部声明了变量tmp,python认为加粗的那一行,就是对函数内tmp的处理,但它写早了,跑到声明前去不合语法;而func1()函数内部没有单独声明一个tmp变量,于是python认为函数处理的是全局变量tmp。
上面乱用了好多术语,都是凭感觉写的!