python的函數(shù)參數(shù)傳遞是"引用傳遞(地址傳遞)"。
成都創(chuàng)新互聯(lián)公司專注于企業(yè)營(yíng)銷型網(wǎng)站建設(shè)、網(wǎng)站重做改版、昌吉網(wǎng)站定制設(shè)計(jì)、自適應(yīng)品牌網(wǎng)站建設(shè)、H5建站、購(gòu)物商城網(wǎng)站建設(shè)、集團(tuán)公司官網(wǎng)建設(shè)、成都外貿(mào)網(wǎng)站建設(shè)、高端網(wǎng)站制作、響應(yīng)式網(wǎng)頁(yè)設(shè)計(jì)等建站業(yè)務(wù),價(jià)格優(yōu)惠性價(jià)比高,為昌吉等各大城市提供網(wǎng)站開發(fā)制作服務(wù)。
python中賦值語(yǔ)句的過程(x = 1):先申請(qǐng)一段內(nèi)存分配給一個(gè)整型對(duì)象來(lái)存儲(chǔ)數(shù)據(jù)1,然后讓變量x去指向這個(gè)對(duì)象,實(shí)際上就是指向這段內(nèi)存(這里有點(diǎn)和C語(yǔ)言中的指針類似)。
在Python中,會(huì)為每個(gè)層次生成一個(gè)符號(hào)表,里層能調(diào)用外層中的變量,而外層不能調(diào)用里層中的變量,并且當(dāng)外層和里層有同名變量時(shí),外層變量會(huì)被里層變量屏蔽掉。函數(shù)? 調(diào)用 ?會(huì)為函數(shù)局部變量生成一個(gè)新的符號(hào)表。
局部變量:作用于該函數(shù)內(nèi)部,一旦函數(shù)執(zhí)行完成,該變量就被回收。
全局變量:它是在函數(shù)外部定義的,作用域是整個(gè)文件。全局變量可以直接在函數(shù)里面應(yīng)用,但是如果要在函數(shù)內(nèi)部改變?nèi)肿兞?,必須使用global關(guān)鍵字進(jìn)行聲明。
注意 :默認(rèn)值在函數(shù)? 定義 ?作用域被解析
在定義函數(shù)時(shí),就已經(jīng)執(zhí)行力它的局部變量
python中不可變類型是共享內(nèi)存地址的:把相同的兩個(gè)不可變類型數(shù)據(jù)賦給兩個(gè)不同變量a,b,a,b在內(nèi)存中的地址是一樣的。
對(duì)象vs變量
在python中,類型屬于對(duì)象,變量是沒有類型的,這正是python的語(yǔ)言特性,也是吸引著很多pythoner的一點(diǎn)。所有的變量都可以理解是內(nèi)存中一個(gè)對(duì)象的“引用”,或者,也可以看似c中void*的感覺。所以,希望大家在看到一個(gè)python變量的時(shí)候,把變量和真正的內(nèi)存對(duì)象分開。
類型是屬于對(duì)象的,而不是變量。
這樣,很多問題就容易思考了。
例如:
對(duì)象vs變量
12
nfoo = 1 #一個(gè)指向int數(shù)據(jù)類型的nfoo(再次提醒,nfoo沒有類型)lstFoo = [1] #一個(gè)指向list類型的lstFoo,這個(gè)list中包含一個(gè)整數(shù)1
可更改(mutable)與不可更改(immutable)對(duì)象
對(duì)應(yīng)于上一個(gè)概念,就必須引出另了另一概念,這就是可更改(mutable)對(duì)象與不可更改(immutable)對(duì)象。
對(duì)于python比較熟悉的人們都應(yīng)該了解這個(gè)事實(shí),在python中,strings, tuples, 和numbers是不可更改的對(duì)象,而list,dict等則是可以修改的對(duì)象。那么,這些所謂的可改變和不可改變影響著什么呢?
可更改vs不可更改
12345
nfoo = 1nfoo = 2lstFoo = [1]lstFoo[0] = 2
代碼第2行中,內(nèi)存中原始的1對(duì)象因?yàn)椴荒芨淖?,于是被“拋棄”,另nfoo指向一個(gè)新的int對(duì)象,其值為2
代碼第5行中,更改list中第一個(gè)元素的值,因?yàn)閘ist是可改變的,所以,第一個(gè)元素變更為2。其實(shí)應(yīng)該說(shuō),lstFoo指向一個(gè)包含一個(gè)對(duì)象的數(shù)組。賦值所發(fā)生的事情,是有一個(gè)新int對(duì)象被指定給lstFoo所指向的數(shù)組對(duì)象的第一個(gè)元素,但是對(duì)于lstFoo本身來(lái)說(shuō),所指向的數(shù)組對(duì)象并沒有變化,只是數(shù)組對(duì)象的內(nèi)容發(fā)生變化了。這個(gè)看似void*的變量所指向的對(duì)象仍舊是剛剛的那個(gè)有一個(gè)int對(duì)象的list。
如下圖所示:
Python的函數(shù)參數(shù)傳遞:傳值?引用?
對(duì)于變量(與對(duì)象相對(duì)的概念),其實(shí),python函數(shù)參數(shù)傳遞可以理解為就是變量傳值操作,用C++的方式理解,就是對(duì)void*賦值。如果這個(gè)變量的值不變,我們看似就是引用,如果這個(gè)變量的值改變,我們看著像是在賦值。有點(diǎn)暈是吧,我們?nèi)耘f據(jù)個(gè)例子。
不可變對(duì)象參數(shù)調(diào)用
12345
def ChangeInt( a ): a = 10nfoo = 2 ChangeInt(nfoo)print nfoo #結(jié)果是2
這時(shí)發(fā)生了什么,有一個(gè)int對(duì)象2,和指向它的變量nfoo,當(dāng)傳遞給ChangeInt的時(shí)候,按照傳值的方式,復(fù)制了變量nfoo的值,這樣,a就是nfoo指向同一個(gè)Int對(duì)象了,函數(shù)中a=10的時(shí)候,發(fā)生什么?(還記得我上面講到的那些概念么),int是不能更改的對(duì)象,于是,做了一個(gè)新的int對(duì)象,另a指向它(但是此時(shí),被變量nfoo指向的對(duì)象,沒有發(fā)生變化),于是在外面的感覺就是函數(shù)沒有改變nfoo的值,看起來(lái)像C++中的傳值方式。
可變對(duì)象參數(shù)調(diào)用
12345
def ChangeList( a ): a[0] = 10lstFoo = [2]ChangeList(lstFoo )print nfoo #結(jié)果是[10]
當(dāng)傳遞給ChangeList的時(shí)候,變量仍舊按照“傳值”的方式,復(fù)制了變量lstFoo 的值,于是a和lstFoo 指向同一個(gè)對(duì)象,但是,list是可以改變的對(duì)象,對(duì)a[0]的操作,就是對(duì)lstFoo指向的對(duì)象的內(nèi)容的操作,于是,這時(shí)的a[0] = 10,就是更改了lstFoo 指向的對(duì)象的第一個(gè)元素,所以,再次輸出lstFoo 時(shí),顯示[10],內(nèi)容被改變了,看起來(lái),像C++中的按引用傳遞。
首先還是應(yīng)該科普下函數(shù)參數(shù)傳遞機(jī)制,傳值和傳引用是什么意思?
函數(shù)參數(shù)傳遞機(jī)制問題在本質(zhì)上是調(diào)用函數(shù)(過程)和被調(diào)用函數(shù)(過程)在調(diào)用發(fā)生時(shí)進(jìn)行通信的方法問題?;镜膮?shù)傳遞機(jī)制有兩種:值傳遞和引用傳遞。
值傳遞(passl-by-value)過程中,被調(diào)函數(shù)的形式參數(shù)作為被調(diào)函數(shù)的局部變量處理,即在堆棧中開辟了內(nèi)存空間以存放由主調(diào)函數(shù)放進(jìn)來(lái)的實(shí)參的值,從而成為了實(shí)參的一個(gè)副本。值傳遞的特點(diǎn)是被調(diào)函數(shù)對(duì)形式參數(shù)的任何操作都是作為局部變量進(jìn)行,不會(huì)影響主調(diào)函數(shù)的實(shí)參變量的值。
引用傳遞(pass-by-reference)過程中,被調(diào)函數(shù)的形式參數(shù)雖然也作為局部變量在堆棧中開辟了內(nèi)存空間,但是這時(shí)存放的是由主調(diào)函數(shù)放進(jìn)來(lái)的實(shí)參變量的地址。被調(diào)函數(shù)對(duì)形參的任何操作都被處理成間接尋址,即通過堆棧中存放的地址訪問主調(diào)函數(shù)中的實(shí)參變量。正因?yàn)槿绱?,被調(diào)函數(shù)對(duì)形參做的任何操作都影響了主調(diào)函數(shù)中的實(shí)參變量。
在python中實(shí)際又是怎么樣的呢?
先看一個(gè)簡(jiǎn)單的例子:
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)
運(yùn)行后結(jié)果如下:
main before invoke test
39601564
test before
39601564
test after +
39601540
main afterf invoke test
2
39601564
id函數(shù)可以獲得對(duì)象的內(nèi)存地址.很明顯從上面例子可以看出,將a變量作為參數(shù)傳遞給了test函數(shù),傳遞了a的一個(gè)引用,把a(bǔ)的地址傳遞過去了,所以在函數(shù)內(nèi)獲取的變量C的地址跟變量a的地址是一樣的,但是在函數(shù)內(nèi),對(duì)C進(jìn)行賦值運(yùn)算,C的值從2變成了4,實(shí)際上2和4所占的內(nèi)存空間都還是存在的,賦值運(yùn)算后,C指向4所在的內(nèi)存。而a仍然指向2所在的內(nèi)存,所以后面打印a,其值還是2.
如果還不能理解,先看下面例子
a=1
b=1
id(a)
40650152
id(b)
40650152
a=2
id(a)
40650140
a和b都是int類型的值,值都是1,而且內(nèi)存地址都是一樣的,這已經(jīng)表明了在python中,可以有多個(gè)引用指向同一個(gè)內(nèi)存(畫了一個(gè)很挫的圖,見諒),在給a賦值為2后,再次查看a的內(nèi)存地址,都已經(jīng)變化了
而基于最前面的例子,大概可以這樣描述:
那python函數(shù)傳參就是傳引用?然后傳參的值在被調(diào)函數(shù)內(nèi)被修改也不影響主調(diào)函數(shù)的實(shí)參變量的值?再來(lái)看個(gè)例子。
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)
實(shí)際值為:
main before invoke test
64129944
test before
64129944
test after +
64129944
main afterf invoke test
['loleina', 30, 'female']
64129944
發(fā)現(xiàn)一樣的傳值,而第二個(gè)變量居然變化,為啥呢?
實(shí)際上是因?yàn)閜ython中的序列:列表是一個(gè)可變的對(duì)象,就基于list1=[1,2] list1[0]=[0]這樣前后的查看list1的內(nèi)存地址,是一樣的。
list1=[1,2]
id(list1)
64185208
list1[0]=[0]
list1
[[0], 2]
id(list1)
64185208
結(jié)論:python不允許程序員選擇采用傳值還是傳引用。Python參數(shù)傳遞采用的肯定是“傳對(duì)象引用”的方式。這種方式相當(dāng)于傳值和傳引用的一種綜合。如果函數(shù)收到的是一個(gè)可變對(duì)象(比如字典或者列表)的引用,就能修改對(duì)象的原始值--相當(dāng)于通過“傳引用”來(lái)傳遞對(duì)象。如果函數(shù)收到的是一個(gè)不可變對(duì)象(比如數(shù)字、字符或者元組)的引用,就不能直接修改原始對(duì)象--相當(dāng)于通過“傳值'來(lái)傳遞對(duì)象。
分類: python 基礎(chǔ)語(yǔ)法
Python中函數(shù)參數(shù)的傳遞是通過“賦值”來(lái)傳遞的。但這條規(guī)則只回答了函數(shù)參數(shù)傳遞的“戰(zhàn)略問題”,并沒有回答“戰(zhàn)術(shù)問題”,也就說(shuō)沒有回答怎么賦值的問題。函數(shù)參數(shù)的使用可以分為兩個(gè)方面,一是函數(shù)參數(shù)如何定義,二是函數(shù)在調(diào)用時(shí)的參數(shù)如何解析的。而后者又是由前者決定的。函數(shù)參數(shù)的定義有四種形式:
1. F(arg1,arg2,...)
2. F(arg2=value,arg3=value...)
3. F(*arg1)
4. F(**arg1)
第1 種方式是最“傳統(tǒng)”的方式:一個(gè)函數(shù)可以定義不限個(gè)數(shù)參數(shù),參數(shù)(形式參數(shù))放在跟在函數(shù)名后面的小括號(hào)中,各個(gè)參數(shù)之間以逗號(hào)隔開。用這種方式定義的函數(shù)在調(diào)用的時(shí)候也必須在函數(shù)名后的小括號(hào)中提供相等個(gè)數(shù)的值(實(shí)際參數(shù)),不能多也不能少,而且順序還必須相同。也就是說(shuō)形參和實(shí)參的個(gè)數(shù)必須一致,而且想給形參1的值必須是實(shí)參中的第一位,形參與實(shí)參之間是一一對(duì)應(yīng)的關(guān)系,即“形參1=實(shí)參1 形參2=實(shí)參2...”。很明顯這是一種非常不靈活的形式。比如:"def addOn(x,y): return x + y",這里定義的函數(shù)addOn,可以用addOn(1,2)的形式調(diào)用,意味著形參x將取值1,主將取值2。addOn(1,2,3)和addOn (1)都是錯(cuò)誤的形式。
第2種方式比第1種方式,在定義的時(shí)候已經(jīng)給各個(gè)形參定義了默認(rèn)值。因此,在調(diào)用這種函數(shù)時(shí),如果沒有給對(duì)應(yīng)的形式參數(shù)傳遞實(shí)參,那么這個(gè)形參就將使用默認(rèn)值。比如:“def addOn(x=3,y=5): return x + y”,那么addOn(6,5)的調(diào)用形式表示形參x取值6,y取值5。此外,addOn(7)這個(gè)形式也是可以的,表示形參x取值7,y取默認(rèn)值5。這時(shí)候會(huì)出現(xiàn)一個(gè)問題,如果想讓x取默認(rèn)值,用實(shí)參給y賦值怎么辦?前面兩種調(diào)用形式明顯就不行了,這時(shí)就要用到Python中函數(shù)調(diào)用方法的另一大絕招 ──關(guān)健字賦值法??梢杂胊ddOn(y=6),這時(shí)表示x取默認(rèn)值3,而y取值6。這種方式通過指定形式參數(shù)可以實(shí)現(xiàn)可以對(duì)形式參數(shù)進(jìn)行“精確攻擊”,一個(gè)副帶的功能是可以不必遵守形式參數(shù)的前后順序,比如:addOn(y=4,x=6),這也是可以的。這種通過形式參數(shù)進(jìn)行定點(diǎn)賦值的方式對(duì)于用第1種方式定義的函數(shù)也是適用的。
上面兩種方式定義的形式參數(shù)的個(gè)數(shù)都是固定的,比如定義函數(shù)的時(shí)候如果定義了5個(gè)形參,那么在調(diào)用的時(shí)候最多也只能給它傳遞5個(gè)實(shí)參。但是在實(shí)際編程中并不能總是確定一個(gè)函數(shù)會(huì)有多少個(gè)參數(shù)。第3種方式就是用來(lái)應(yīng)對(duì)這種情況的。它以一個(gè)*加上形參名的方式表示,這個(gè)函數(shù)實(shí)際參數(shù)是不一定的,可以是零個(gè),也可以是N個(gè)。不管是多少個(gè),在函數(shù)內(nèi)部都被存放在以形參名為標(biāo)識(shí)符的tuple中。比如:
對(duì)這個(gè)函數(shù)的調(diào)用addOn() addOn(2) addOn(3,4,5,6)等等都是可以的。
與第3種方式類似,形參名前面加了兩個(gè)*表示,參數(shù)在函數(shù)內(nèi)部將被存放在以形式名為標(biāo)識(shí)符的dictionary中。這時(shí)候調(diào)用函數(shù)必須采用key1=value1、key2=value2...的形式。比如:
1. def addOn(**arg):
2. sum = 0
3. if len(arg) == 0: return 0
4. else:
5. for x in arg.itervalues():
6. sum += x
7. return sum
那么對(duì)這個(gè)函數(shù)的調(diào)用可以用addOn()或諸如addOn(x=4,y=5,k=6)等的方式調(diào)用。
上面說(shuō)了四種函數(shù)形式定義的方式以及他們的調(diào)用方式,是分開說(shuō)的,其實(shí)這四種方式可以組合在一起形成復(fù)雜多樣的形參定義形式。在定義或調(diào)用這種函數(shù)時(shí),要遵循以下規(guī)則:
1. arg=value必須在arg后
2. *arg必須在arg=value后
3. **arg必須在*arg后
在函數(shù)調(diào)用過程中,形參賦值的過程是這樣的:
首先按順序把“arg”這種形式的實(shí)參給對(duì)應(yīng)的形參
第二,把“arg=value”這種形式的實(shí)參賦值給形式
第三,把多出來(lái)的“arg”這種形式的實(shí)參組成一個(gè)tuple給帶一個(gè)星號(hào)的形參
第四,把多出來(lái)的“key=value”這種形式的實(shí)參轉(zhuǎn)為一個(gè)dictionary給帶兩個(gè)星號(hào)的形參。
例子:
1. def test(x,y=5,*a,**b):
2. print x,y,a,b
就這么一個(gè)簡(jiǎn)單函數(shù),來(lái)看看下面對(duì)這個(gè)函數(shù)調(diào)用會(huì)產(chǎn)生什么結(jié)果:
test(1) === 1 5 () {}
test(1,2) === 1 2 () {}
test(1,2,3) === 1 2 (3,) {}
test(1,2,3,4) === 1 2 (3,4)
test(x=1) === 1 5 () {}
test(x=1,y=1) === 1 1 () {}
test(x=1,y=1,a=1) === 1 1 () {'a':1}
test(x=1,y=1,a=1,b=1) === 1 1 () {'a':1,'b':1}
test(1,y=1) === 1 1 () {}
test(1,2,y=1) === 出錯(cuò),說(shuō)y給賦了多個(gè)值
test(1,2,3,4,a=1) === 1 2 (3,4) {'a':1}
test(1,2,3,4,k=1,t=2,o=3) === 1 2 (3,4) {'k':1,'t':2,'o':3}
python將值傳遞參數(shù)的方法:
將值賦給變量url,然后調(diào)用函數(shù),將url寫到函數(shù)名后面的括號(hào)中,這樣就可以將值傳遞給函數(shù)的參數(shù)y了
示例代碼如下:
執(zhí)行結(jié)果如下:
更多Python知識(shí),請(qǐng)關(guān)注:Python自學(xué)網(wǎng)??!