詳解python函數(shù)傳參是傳值還是傳引用
首先還是應(yīng)該科普下函數(shù)參數(shù)傳遞機制,傳值和傳引用是什么意思?
函數(shù)參數(shù)傳遞機制問題在本質(zhì)上是調(diào)用函數(shù)(過程)和被調(diào)用函數(shù)(過程)在調(diào)用發(fā)生時進行通信的方法問題?;镜膮?shù)傳遞機制有兩種:值傳遞和引用傳遞。
值傳遞(passl-by-value)過程中,被調(diào)函數(shù)的形式參數(shù)作為被調(diào)函數(shù)的局部變量處理,即在堆棧中開辟了內(nèi)存空間以存放由主調(diào)函數(shù)放進來的實參的值,從而成為了實參的一個副本。值傳遞的特點是被調(diào)函數(shù)對形式參數(shù)的任何操作都是作為局部變量進行,不會影響主調(diào)函數(shù)的實參變量的值。
引用傳遞(pass-by-reference)過程中,被調(diào)函數(shù)的形式參數(shù)雖然也作為局部變量在堆棧中開辟了內(nèi)存空間,但是這時存放的是由主調(diào)函數(shù)放進來的實參變量的地址。被調(diào)函數(shù)對形參的任何操作都被處理成間接尋址,即通過堆棧中存放的地址訪問主調(diào)函數(shù)中的實參變量。正因為如此,被調(diào)函數(shù)對形參做的任何操作都影響了主調(diào)函數(shù)中的實參變量。
在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
def 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)
運行后結(jié)果如下:
>>> main before invoke test 39601564 test before 39601564 test after + 39601540 main afterf invoke test 2 39601564
id函數(shù)可以獲得對象的內(nèi)存地址.很明顯從上面例子可以看出,將a變量作為參數(shù)傳遞給了test函數(shù),傳遞了a的一個引用,把a的地址傳遞過去了,所以在函數(shù)內(nèi)獲取的變量C的地址跟變量a的地址是一樣的,但是在函數(shù)內(nèi),對C進行賦值運算,C的值從2變成了4,實際上2和4所占的內(nèi)存空間都還是存在的,賦值運算后,C指向4所在的內(nèi)存。而a仍然指向2所在的內(nèi)存,所以后面打印a,其值還是2.
如果還不能理解,先看下面例子
>>> a=1 >>> b=1 >>> id(a) 40650152 >>> id(b) 40650152 >>> a=2 >>> id(a) 40650140
a和b都是int類型的值,值都是1,而且內(nèi)存地址都是一樣的,這已經(jīng)表明了在python中,可以有多個引用指向同一個內(nèi)存(畫了一個很挫的圖,見諒),在給a賦值為2后,再次查看a的內(nèi)存地址,都已經(jīng)變化了
而基于最前面的例子,大概可以這樣描述:
那python函數(shù)傳參就是傳引用?然后傳參的值在被調(diào)函數(shù)內(nèi)被修改也不影響主調(diào)函數(shù)的實參變量的值?再來看個例子。
from ctypes import *
import os.path
import sys
def test(list2):
print "test before "
print id(list2)
list2[1]=30
print "test after +"
print id(list2)
return list2
def 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 test 64129944 test before 64129944 test after + 64129944 main afterf invoke test ['loleina', 30, 'female'] 64129944
發(fā)現(xiàn)一樣的傳值,而第二個變量居然變化,為啥呢?
實際上是因為python中的序列:列表是一個可變的對象,就基于list1=[1,2] list1[0]=[0]這樣前后的查看list1的內(nèi)存地址,是一樣的。
>>> list1=[1,2] >>> id(list1) 64185208 >>> list1[0]=[0] >>> list1 [[0], 2] >>> id(list1) 64185208
結(jié)論:python不允許程序員選擇采用傳值還是傳引用。Python參數(shù)傳遞采用的肯定是“傳對象引用”的方式。這種方式相當于傳值和傳引用的一種綜合。如果函數(shù)收到的是一個可變對象(比如字典或者列表)的引用,就能修改對象的原始值--相當于通過“傳引用”來傳遞對象。如果函數(shù)收到的是一個不可變對象(比如數(shù)字、字符或者元組)的引用,就不能直接修改原始對象--相當于通過“傳值'來傳遞對象。
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

