国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Python > 正文

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

2019-11-09 20:24:53
字體:
來源:轉載
供稿:網友

python不允許程序員選擇采用傳值還是傳引用。Python參數傳遞采用的肯定是“傳對象引用”的方式:如果函數收到的是一個可變對象(比如字典或者列表)的引用,就能修改對象的原始值--相當于通過“傳引用”來傳遞對象。如果函數收到的是一個不可變對象(比如數字、字符或者元組)的引用,就不能直接修改原始對象--相當于通過“傳值'來傳遞對象。

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

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

先看一個簡單的例子:

   

復制代碼
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 test39601564test before 39601564test after +39601540main afterf invoke test239601564復制代碼

 

 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 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 test64129944test before 64129944test after +64129944main 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參數傳遞采用的肯定是“傳對象引用”的方式。這種方式相當于傳值和傳引用的一種綜合。如果函數收到的是一個可變對象(比如字典或者列表)的引用,就能修改對象的原始值--相當于通過“傳引用”來傳遞對象。如果函數收到的是一個不可變對象(比如數字、字符或者元組)的引用,就不能直接修改原始對象--相當于通過“傳值'來傳遞對象。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 井陉县| 双桥区| 张家口市| 南华县| 大庆市| 莎车县| 南乐县| 始兴县| 蒙自县| 遂平县| 桦甸市| 原平市| 隆尧县| 平利县| 白河县| 福清市| 云和县| 宁武县| 盐边县| 科技| 宁乡县| 明溪县| 韶关市| 城步| 石景山区| 禹州市| 绥德县| 康马县| 门源| 文安县| 武定县| 云安县| 永顺县| 泰宁县| 施甸县| 甘南县| 新田县| 修文县| 宝应县| 蓬莱市| 虹口区|