探秘Python里的函數(shù)參數(shù)傳遞方式
原創(chuàng)?xinxin?菜鳥學(xué)Python?
閱讀本文大概需要6分鐘
經(jīng)常用python函數(shù)的同學(xué)們可能會(huì)有一個(gè)疑問窥翩,Python的函數(shù)的入?yún)⒌降资莻髦的剡€是引用,其實(shí)說實(shí)話褂痰,這個(gè)問題在我剛開始學(xué)python的時(shí)候也糾結(jié)過势就,因?yàn)閷?duì)于 c/c++,里面函數(shù)參數(shù)傳遞是有傳地址也有傳值的辫红。所以在python里面到底又是怎么樣.
其實(shí)這個(gè)問題很有意思怕轿,把這個(gè)問題弄明白之后肄满,對(duì)python的理解又前進(jìn)了一步.
開始之前我們先說一下什么是傳值,什么是傳引用
傳值
簡(jiǎn)單來說智政,你在內(nèi)存中有一個(gè)地址认罩,我也有一個(gè)地址,我呢把我的地址里面的內(nèi)容復(fù)制給你续捂,以后你做什么就跟我沒有啥關(guān)系垦垂,咱倆井水不犯河水。不會(huì)改變?cè)瓉淼膮?shù)的內(nèi)容
傳引用
所謂傳引用是有一個(gè)參數(shù)在內(nèi)存有個(gè)地址牙瓢,地址里面放了一堆東西,在調(diào)用函數(shù)時(shí)劫拗,把實(shí)際參數(shù)的地址傳遞到函數(shù)中,那么在函數(shù)中對(duì)參數(shù)所進(jìn)行的修改矾克,將影響到實(shí)際參數(shù)页慷。也就是說最后函數(shù)運(yùn)行完之后會(huì)改變?cè)瓉淼膮?shù)的內(nèi)容.
好下面我們來說一說Python里面是怎么做的,希望能給初學(xué)者一些啟示:
1.先看一個(gè)簡(jiǎn)單的例子
例1:傳引用
如果是傳引用胁附,也就是說最后會(huì)改變?cè)瓉淼膮?shù)酒繁,那么n的值在函數(shù)內(nèi)修改了變成了11,那么n最后應(yīng)該是11啊控妻,但是n還是10州袒,沒有變化,看來python不是傳引用弓候,那哎哎哎郎哭,是不是傳值呢,好我們接著往下探.
2.再看一個(gè)小例子
例2:傳值
若是傳值弓叛,old原來是['a','b','c']那么運(yùn)行完change()函數(shù)之后彰居,應(yīng)該還是['a','b','c'],顯然最后old的內(nèi)容改變了诚纸。
那么python中的參數(shù)傳遞也不是傳值撰筷,問題來了不是傳引用也不是傳值,到底是傳啥呢~~
3.深入探究
????????a=1,b=a,b=2
這個(gè)在C/C++里面執(zhí)行b=a的時(shí)候畦徘,其實(shí)是在內(nèi)存里面申請(qǐng)一塊內(nèi)存把a(bǔ)的值復(fù)制到內(nèi)存中毕籽,當(dāng)執(zhí)行b=2的時(shí)候,是把b對(duì)應(yīng)的值從1修改成2,如圖:
但是在Python里面的賦值并不是復(fù)制井辆,b=a的操作使得b與a引用同一個(gè)對(duì)象(注意是對(duì)象),而b=2則是b指向2关筒,如圖:
不信我們可以在程序里面驗(yàn)證:
a=1
print(id(a))
>>>33989408
b=a
print(id(b))?#b=a之后b的id()值和a一樣
>>>33989408
b=2
print(id(b))
>>>33989396?#b=2之后b指向?qū)ο?,id()值發(fā)生了改變
print(id(a))
>>>33989408
從程序里面寫可以看出
b=a賦值后b的id()和a一樣,b=2之后b指向了另外一塊空間杯缺,其實(shí)b=a傳遞的是對(duì)象的引用蒸播,它們指向同一塊內(nèi)存.
當(dāng)b=2之后b又重新指向了2所代表的對(duì)象上去,而此時(shí)1只有a指向.
4.揭開謎底
說了一大圈是時(shí)候揭開底牌了.對(duì)于例1,n=n+1袍榆,因?yàn)閚是數(shù)字,是不可變的對(duì)象胀屿,n+1會(huì)重新申請(qǐng)一塊新的內(nèi)存,其值為n+1,并在函數(shù)體中創(chuàng)建局部變量n指向它包雀。當(dāng)函數(shù)add()調(diào)用完之后宿崭,函數(shù)體中的局部變量在函數(shù)體外不可見,此時(shí)的n代表函數(shù)體外的命名空間所對(duì)應(yīng)的n才写,所以其值還是10.
結(jié)論:Python函數(shù)參數(shù)傳遞的是對(duì)象的引用,參數(shù)傳遞的過程中將整個(gè)對(duì)象傳入.
對(duì)于可變的對(duì)象的修改在函數(shù)外部和內(nèi)部都可見葡兑,調(diào)用者和被調(diào)用者共享這個(gè)對(duì)象
而對(duì)于不可變對(duì)象,由于并不能真正被修改赞草,因?yàn)樾薷囊话愣际峭ㄟ^生成一個(gè)新的對(duì)象然后賦值來實(shí)現(xiàn)的讹堤。
(好謎底揭開了,剛才我特意沒有解釋例2厨疙,現(xiàn)在你試著去理解看看蜕劝,是不是能明白.)
好了最后說一下,若我寫的對(duì)大家有幫助轰异,麻煩大家支持一下,也是對(duì)我的一點(diǎn)鼓勵(lì)和動(dòng)力.