python函數參數引用傳遞

 首先還是應該科普下函數參數傳遞機制傳值和傳引用是什麼意思?

   函數參數傳遞機制問題在本質上是調用函數(過程)和被調用函數(過程)在調用發生時進行通信的方法問題。基本的參數傳遞機制有兩種:值傳遞和引用傳遞。

  值傳遞(passl-by-value)過程中,被調函數的形式參數作爲被調函數的局部變量處理,即在堆棧中開闢了內存空間以存放由主調函數放進來的實參的值,從而成爲了實參的一個副本。值傳遞的特點是被調函數對形式參數的任何操作都是作爲局部變量進行,不會影響主調函數的實參變量的值。

  引用傳遞(pass-by-reference)過程中,被調函數的形式參數雖然也作爲局部變量在堆棧中開闢了內存空間,但是這時存放的是由主調函數放進來的實參變量的地址。被調函數對形參的任何操作都被處理成間接尋址,即通過堆棧中存放的地址訪問主調函數中的實參變量。正因爲如此,被調函數對形參做的任何操作都影響了主調函數中的實參變量。

      在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)
    
複製代碼

運行後結果如下:

 

複製代碼
>>> 
main before invoke test
39601564
test before 
39601564
test after +
39601540
main afterf invoke test
2
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)
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 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
複製代碼

      發現一樣的傳值,而第二個變量居然變化,爲啥呢?

      實際上是因爲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參數傳遞採用的肯定是“傳對象引用”的方式。這種方式相當於傳值和傳引用的一種綜合。如果函數收到的是一個可變對象(比如字典或者列表)的引用,就能修改對象的原始值--相當於通過“傳引用”來傳遞對象。如果函數收到的是一個不可變對象(比如數字、字符或者元組)的引用,就不能直接修改原始對象--相當於通過“傳值'來傳遞對象。


文章引用:

http://www.cnblogs.com/loleina/p/5276918.html


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章