標籤:實參 不能 程式員 ide == display range image from
首先還是應該科普下函數參數傳遞機制,傳值和傳引用是什麼意思?
函數參數傳遞機制問題在本質上是調用函數(過程)和被調用函數(過程)在調用發生時進行通訊的方法問題。基本的參數傳遞機制有兩種:值傳遞和引用傳遞。
值傳遞(passl-by-value)過程中,被調函數的形式參數作為被調函數的局部變數處理,即在堆棧中開闢了記憶體空間以存放由主調函數放進來的實參的值,從而成為了實參的一個副本。值傳遞的特點是被調函數對形式參數的任何操作都是作為局部變數進行,不會影響主調函數的實參變數的值。
引用傳遞(pass-by-reference)過程中,被調函數的形式參數雖然也作為局部變數在堆棧中開闢了記憶體空間,但是這時存放的是由主調函數放進來的實參變數的地址。被調函數對形參的任何操作都被處理成間接定址,即通過堆棧中存放的地址訪問主調函數中的實參變數。正因為如此,被調函數對形參做的任何操作都影響了主調函數中的實參變數。
在python中實際又是怎麼樣的呢?
先看一個簡單的例子:
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)
View Code
運行後結果如下:
>>>
main before invoke test
test before
test after +
main afterf invoke test
39601564
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的記憶體位址,都已經變化了
而基於最前面的例子,大概可以這樣描述:
那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)
View Code
實際值為:
>>>
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)
結論:python不允許程式員選擇採用傳值還是傳引用。Python參數傳遞採用的肯定是“傳對象引用”的方式。這種方式相當於傳值和傳引用的一種綜合。如果函數收到的是一個可變對象(比如字典或者列表)的引用,就能修改對象的原始值--相當於通過“傳引用”來傳遞對象。如果函數收到的是一個不可變對象(比如數字、字元或者元組)的引用,就不能直接修改原始對象--相當於通過“傳值‘來傳遞對象。
轉自:http://www.cnblogs.com/loleina/p/5276918.html
[轉]Python中函數的值傳遞和引用傳遞