python函數傳參是傳值仍是傳引用?

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

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

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

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

      在python中實際又是怎麼樣的呢?code

      先看一個簡單的例子:對象

   

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)
    

運行後結果以下:blog

 

>>> 
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.內存

   若是還不能理解,先看下面例子ssl

>>> a=1
>>> b=1
>>> id(a)
40650152
>>> id(b)
40650152
>>> a=2
>>> id(a)
40650140class

       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參數傳遞採用的確定是「傳對象引用」的方式。這種方式至關於傳值和傳引用的一種綜合。若是函數收到的是一個可變對象(好比字典或者列表)的引用,就能修改對象的原始值--至關於經過「傳引用」來傳遞對象。若是函數收到的是一個不可變對象(好比數字、字符或者元組)的引用,就不能直接修改原始對象--至關於經過「傳值'來傳遞對象。

相關文章
相關標籤/搜索