您的位置:首頁技術文章
文章詳情頁

Python參數傳遞機制傳值和傳引用原理詳解

瀏覽:101日期:2022-07-24 18:30:50

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

函數參數傳遞機制問題在本質上是調用函數(過程)和被調用函數(過程)在調用發生時進行通信的方法問題?;镜膮祩鬟f機制有兩種:值傳遞和引用傳遞。

值傳遞(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)

運行后結果如下:

>>> main before invoke testtest before test after +main afterf invoke test39601564

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參數傳遞機制傳值和傳引用原理詳解

而基于最前面的例子,大概可以這樣描述:

Python參數傳遞機制傳值和傳引用原理詳解

那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)

實際值為:

>>> main before invoke testtest 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)

字典也是可變對象:

>>> def fun2(num1,l1,d1):... num1=123... l1[0]=123... d1[’a’]=123... print('inside:','num1=%f,l1=%s,d1=%s'%(num1,l1,d1))... >>> num=111>>> l=[1,1,1]>>> d={’a’:111,’b’:0}>>> print('before:','num=%f,l=%s,d=%s'%(num,l,d))before: num=111.000000,l=[1, 1, 1],d={’a’: 111, ’b’: 0}>>> fun2(num,l,d)inside: num1=123.000000,l1=[123, 1, 1],d1={’a’: 123, ’b’: 0}>>> print('after:','num=%f,l=%s,d=%s'%(num,l,d))after: num=111.000000,l=[123, 1, 1],d={’a’: 123, ’b’: 0}

結論:python不允許程序員選擇采用傳值還是傳引用。Python參數傳遞采用的肯定是“傳對象引用”的方式。這種方式相當于傳值和傳引用的一種綜合。

如果函數收到的是一個可變對象(比如字典或者列表)的引用,就能修改對象的原始值--相當于通過“傳引用”來傳遞對象。

如果函數收到的是一個不可變對象(比如數字、字符或者元組)的引用,就不能直接修改原始對象--相當于通過“傳值’來傳遞對象。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Python 編程
相關文章:
国产综合久久一区二区三区