python函數(shù)參數(shù)原理 python函數(shù)做參數(shù)

Python 函數(shù)中,參數(shù)是傳值,還是傳引用

首先還是應(yīng)該科普下函數(shù)參數(shù)傳遞機(jī)制,傳值和傳引用是什么意思?

創(chuàng)新互聯(lián)公司作為成都網(wǎng)站建設(shè)公司,專(zhuān)注成都網(wǎng)站建設(shè)、網(wǎng)站設(shè)計(jì),有關(guān)企業(yè)網(wǎng)站建設(shè)方案、改版、費(fèi)用等問(wèn)題,行業(yè)涉及混凝土攪拌罐等多個(gè)領(lǐng)域,已為上千家企業(yè)服務(wù),得到了客戶(hù)的尊重與認(rèn)可。

函數(shù)參數(shù)傳遞機(jī)制問(wèn)題在本質(zhì)上是調(diào)用函數(shù)(過(guò)程)和被調(diào)用函數(shù)(過(guò)程)在調(diào)用發(fā)生時(shí)進(jìn)行通信的方法問(wèn)題?;镜膮?shù)傳遞機(jī)制有兩種:值傳遞和引用傳遞。

值傳遞(passl-by-value)過(guò)程中,被調(diào)函數(shù)的形式參數(shù)作為被調(diào)函數(shù)的局部變量處理,即在堆棧中開(kāi)辟了內(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)過(guò)程中,被調(diào)函數(shù)的形式參數(shù)雖然也作為局部變量在堆棧中開(kāi)辟了內(nèi)存空間,但是這時(shí)存放的是由主調(diào)函數(shù)放進(jìn)來(lái)的實(shí)參變量的地址。被調(diào)函數(shù)對(duì)形參的任何操作都被處理成間接尋址,即通過(guò)堆棧中存放的地址訪問(wèn)主調(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ǔ)的地址傳遞過(guò)去了,所以在函數(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類(lèi)型的值,值都是1,而且內(nèi)存地址都是一樣的,這已經(jīng)表明了在python中,可以有多個(gè)引用指向同一個(gè)內(nèi)存(畫(huà)了一個(gè)很挫的圖,見(jiàn)諒),在給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)于通過(guò)“傳引用”來(lái)傳遞對(duì)象。如果函數(shù)收到的是一個(gè)不可變對(duì)象(比如數(shù)字、字符或者元組)的引用,就不能直接修改原始對(duì)象--相當(dāng)于通過(guò)“傳值'來(lái)傳遞對(duì)象。

分類(lèi): python 基礎(chǔ)語(yǔ)法

python函數(shù)調(diào)用的參數(shù)傳遞

python的函數(shù)參數(shù)傳遞是"引用傳遞(地址傳遞)"。

python中賦值語(yǔ)句的過(guò)程(x = 1):先申請(qǐng)一段內(nèi)存分配給一個(gè)整型對(duì)象來(lái)存儲(chǔ)數(shù)據(jù)1,然后讓變量x去指向這個(gè)對(duì)象,實(shí)際上就是指向這段內(nèi)存(這里有點(diǎn)和C語(yǔ)言中的指針類(lèi)似)。

在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中不可變類(lèi)型是共享內(nèi)存地址的:把相同的兩個(gè)不可變類(lèi)型數(shù)據(jù)賦給兩個(gè)不同變量a,b,a,b在內(nèi)存中的地址是一樣的。

Python的位置參數(shù)、默認(rèn)參數(shù)、關(guān)鍵字參數(shù)、可變參數(shù)區(qū)別

對(duì)于python函數(shù)參數(shù),對(duì)于初學(xué)者可能就是進(jìn)入了迷宮,盡管我也是初學(xué)者,簡(jiǎn)單總結(jié)一下。

說(shuō)參數(shù)之前,先講一下兩個(gè)packing(包裹)和unpacking(解包裹):

輸出:

我總結(jié)不了這個(gè)概念,只能幫大家到這了

一、位置參數(shù)和關(guān)鍵字參數(shù):

調(diào)用函數(shù)時(shí)根據(jù)函數(shù)定義的參數(shù)位置來(lái)傳遞參數(shù)。

注意:

有位置參數(shù)時(shí),位置參數(shù)必須在關(guān)鍵字參數(shù)的前面,但關(guān)鍵字參數(shù)之間不存在先后順序的

二、默認(rèn)參數(shù):

用于定義函數(shù),為參數(shù)提供默認(rèn)值,調(diào)用函數(shù)時(shí)可傳可不傳該默認(rèn)參數(shù)的值(注意:所有位置參數(shù)必須出現(xiàn)在默認(rèn)參數(shù)前,包括函數(shù)定義和調(diào)用)

三、可變參數(shù):

定義函數(shù)時(shí),有時(shí)候我們不確定調(diào)用的時(shí)候會(huì)傳遞多少個(gè)參數(shù)(不傳參也可以)。此時(shí),可用包裹(packing)位置參數(shù),或者包裹關(guān)鍵字參數(shù),來(lái)進(jìn)行參數(shù)傳遞,會(huì)顯得非常方便。

1、包裹位置傳遞

我們傳進(jìn)的所有參數(shù)都會(huì)被args變量收集,它會(huì)根據(jù)傳進(jìn)參數(shù)的位置合并為一個(gè)元組(tuple),args是元組類(lèi)型,這就是包位置傳遞。

2、包裹關(guān)鍵字傳遞

kargs是一個(gè)字典(dict),收集所有關(guān)鍵字參數(shù)

四、解包裹參數(shù):

*args 和 **kargs ,也可以在函數(shù)調(diào)用的時(shí)候使用,稱(chēng)之為解包(unpacking)

1、在傳遞元組時(shí),讓元組的每一個(gè)元素對(duì)應(yīng)一個(gè)位置參數(shù)

2、在傳遞詞典字典時(shí),讓詞典的每個(gè)鍵值對(duì)作為一個(gè)關(guān)鍵字參數(shù)傳遞給函數(shù)

五、位置參數(shù)、默認(rèn)參數(shù)、可變參數(shù)的混合使用

1、基本原則是:先位置參數(shù),默認(rèn)參數(shù),包裹位置,包裹關(guān)鍵字(定義和調(diào)用都應(yīng)遵循)

2、Python中 *args 和 **kwargs 的區(qū)別

先看個(gè)demo:

輸出結(jié)果:

分析一下:可以看到,這兩個(gè)是[Python]中的可變參數(shù)。 *args 表示任何多個(gè)無(wú)名參數(shù),它是一個(gè)tuple; **kwargs 表示關(guān)鍵字參數(shù),它是一個(gè)dict。并且同時(shí)使用 *args 和 **kwargs 時(shí),必須 *args 參數(shù)列要在 **kwargs 前,否則會(huì)報(bào)語(yǔ)法錯(cuò)誤?。。?/p>

還有個(gè)小應(yīng)用場(chǎng)景:創(chuàng)建字典

其實(shí)python中就帶有dict類(lèi),使用dict(a=1,b=2,c=3)即可創(chuàng)建一個(gè)字典了。

*args:

重點(diǎn)在*,后面的args相當(dāng)于一個(gè)變量名,可以自己定義的。它的本質(zhì)就是將標(biāo)準(zhǔn)調(diào)用剩下的值集中轉(zhuǎn)變?yōu)樵M。

從形參的角度:

從實(shí)參的角度:

從不同角度看**kwargs:

**kwargs與位置參數(shù)和默認(rèn)參數(shù)混用:

超復(fù)雜混合參數(shù)混用記:

總結(jié):

位置參數(shù):

調(diào)用函數(shù)時(shí)所傳參數(shù)的位置必須與定義函數(shù)時(shí)參數(shù)的位置相同

關(guān)鍵字參數(shù):

使用關(guān)鍵字參數(shù)會(huì)指定參數(shù)值賦給哪個(gè)形參,調(diào)用時(shí)所傳參數(shù)的位置可以任意

*位置參數(shù):可接受任意數(shù)量的位置參數(shù)(元組);只能作為最后一個(gè)位置參數(shù)出現(xiàn),其后參數(shù)均為關(guān)鍵字參數(shù)

**關(guān)鍵字參數(shù):可接受任意數(shù)量的關(guān)鍵字參數(shù)(字典);只能作為最后一個(gè)參數(shù)出現(xiàn)

Python 的函數(shù)是怎么傳遞參數(shù)的

首先你要明白,Python的函數(shù)傳遞方式是賦值,而賦值是通過(guò)建立變量與對(duì)象的關(guān)聯(lián)實(shí)現(xiàn)的。

對(duì)于你的代碼:

執(zhí)行 d = 2時(shí),你在__main__里創(chuàng)建了d,并讓它指向2這個(gè)整型對(duì)象。

執(zhí)行函數(shù)add(d)過(guò)程中:

d被傳遞給add()函數(shù)后,在函數(shù)內(nèi)部,num也指向了__main__中的2

但執(zhí)行num = num + 10之后,新建了對(duì)象12,并讓num指向了這個(gè)新對(duì)象——12。

如果你明白函數(shù)中的局部變量與__main__中變量的區(qū)別,那么很顯然,在__main__中,d仍在指著2這個(gè)對(duì)象,它沒(méi)有改變。因此,你打印d時(shí)得到了2。

如果你想讓輸出為12,最簡(jiǎn)潔的辦法是:

在函數(shù)add()里增加return num

調(diào)用函數(shù)時(shí)使用d = add(d)

代碼如下:

def add(num):

num += 10

return num

d = 2

d = add(d)

print d

Python函數(shù)的參數(shù)類(lèi)型

Python函數(shù)的參數(shù)類(lèi)型主要包括必選參數(shù)、可選參數(shù)、可變參數(shù)、位置參數(shù)和關(guān)鍵字參數(shù),本文介紹一下他們的定義以及可變數(shù)據(jù)類(lèi)型參數(shù)傳遞需要注意的地方。

必選參數(shù)(Required arguments)是必須輸入的參數(shù),比如下面的代碼,必須輸入2個(gè)參數(shù),否則就會(huì)報(bào)錯(cuò):

其實(shí)上面例子中的參數(shù) num1和num2也屬于關(guān)鍵字參數(shù),比如可以通過(guò)如下方式調(diào)用:

執(zhí)行結(jié)果:

可選參數(shù)(Optional arguments)可以不用傳入函數(shù),有一個(gè)默認(rèn)值,如果沒(méi)有傳入會(huì)使用默認(rèn)值,不會(huì)報(bào)錯(cuò)。

位置參數(shù)(positional arguments)根據(jù)其在函數(shù)定義中的位置調(diào)用,下面是pow()函數(shù)的幫助信息:

x,y,z三個(gè)參數(shù)的的順序是固定的,并且不能使用關(guān)鍵字:

輸出:

在上面的pow()函數(shù)幫助信息中可以看到位置參數(shù)后面加了一個(gè)反斜杠 / ,這是python內(nèi)置函數(shù)的語(yǔ)法定義,Python開(kāi)發(fā)人員不能在python3.8版本之前的代碼中使用此語(yǔ)法。但python3.0到3.7版本可以使用如下方式定義位置參數(shù):

星號(hào)前面的參數(shù)為位置參數(shù)或者關(guān)鍵字參數(shù),星號(hào)后面是強(qiáng)制關(guān)鍵字參數(shù),具體介紹見(jiàn)強(qiáng)制關(guān)鍵字參數(shù)。

python3.8版本引入了強(qiáng)制位置參數(shù)(Positional-Only Parameters),也就是我們可以使用反斜杠 / 語(yǔ)法來(lái)定義位置參數(shù)了,可以寫(xiě)成如下形式:

來(lái)看下面的例子:

python3.8運(yùn)行:

不能使用關(guān)鍵字參數(shù)形式賦值了。

可變參數(shù) (varargs argument) 就是傳入的參數(shù)個(gè)數(shù)是可變的,可以是0-n個(gè),使用星號(hào)( * )將輸入?yún)?shù)自動(dòng)組裝為一個(gè)元組(tuple):

執(zhí)行結(jié)果:

關(guān)鍵字參數(shù)(keyword argument)允許將任意個(gè)含參數(shù)名的參數(shù)導(dǎo)入到python函數(shù)中,使用雙星號(hào)( ** ),在函數(shù)內(nèi)部自動(dòng)組裝為一個(gè)字典。

執(zhí)行結(jié)果:

上面介紹的參數(shù)可以混合使用:

結(jié)果:

注意:由于傳入的參數(shù)個(gè)數(shù)不定,所以當(dāng)與普通參數(shù)一同使用時(shí),必須把帶星號(hào)的參數(shù)放在最后。

強(qiáng)制關(guān)鍵字參數(shù)(Keyword-Only Arguments)是python3引入的特性,可參考:。 使用一個(gè)星號(hào)隔開(kāi):

在位置參數(shù)一節(jié)介紹過(guò)星號(hào)前面的參數(shù)可以是位置參數(shù)和關(guān)鍵字參數(shù)。星號(hào)后面的參數(shù)都是強(qiáng)制關(guān)鍵字參數(shù),必須以指定參數(shù)名的方式傳參,如果強(qiáng)制關(guān)鍵字參數(shù)沒(méi)有設(shè)置默認(rèn)參數(shù),調(diào)用函數(shù)時(shí)必須傳參。

執(zhí)行結(jié)果:

也可以在可變參數(shù)后面命名關(guān)鍵字參數(shù),這樣就不需要星號(hào)分隔符了:

執(zhí)行結(jié)果:

在Python對(duì)象及內(nèi)存管理機(jī)制中介紹了python中的參數(shù)傳遞屬于對(duì)象的 引用傳遞 (pass by object reference),在編寫(xiě)函數(shù)的時(shí)候需要特別注意。

先來(lái)看個(gè)例子:

執(zhí)行結(jié)果:

l1 和 l2指向相同的地址,由于列表可變,l1改變時(shí),l2也跟著變了。

接著看下面的例子:

結(jié)果:

l1沒(méi)有變化!為什么不是[1, 2, 3, 4]呢?

l = l + [4]表示創(chuàng)建一個(gè)“末尾加入元素 4“的新列表,并讓 l 指向這個(gè)新的對(duì)象,l1沒(méi)有進(jìn)行任何操作,因此 l1 的值不變。如果要改變l1的值,需要加一個(gè)返回值:

結(jié)果:

下面的代碼執(zhí)行結(jié)果又是什么呢?

執(zhí)行結(jié)果:

和第一個(gè)例子一樣,l1 和 l2指向相同的地址,所以會(huì)一起改變。這個(gè)問(wèn)題怎么解決呢?

可以使用下面的方式:

也可以使用淺拷貝或者深度拷貝,具體使用方法可參考Python對(duì)象及內(nèi)存管理機(jī)制。這個(gè)問(wèn)題在Python編程時(shí)需要特別注意。

本文主要介紹了python函數(shù)的幾種參數(shù)類(lèi)型:必選參數(shù)、可選參數(shù)、可變參數(shù)、位置參數(shù)、強(qiáng)制位置參數(shù)、關(guān)鍵字參數(shù)、強(qiáng)制關(guān)鍵字參數(shù),注意他們不是完全獨(dú)立的,比如必選參數(shù)、可選參數(shù)也可以是關(guān)鍵字參數(shù),位置參數(shù)可以是必選參數(shù)或者可選參數(shù)。

另外,python中的參數(shù)傳遞屬于對(duì)象的 引用傳遞 ,在對(duì)可變數(shù)據(jù)類(lèi)型進(jìn)行參數(shù)傳遞時(shí)需要特別注意,如有必要,使用python的拷貝方法。

參考文檔:

--THE END--

當(dāng)前題目:python函數(shù)參數(shù)原理 python函數(shù)做參數(shù)
網(wǎng)頁(yè)地址:http://muchs.cn/article12/doccodc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供App設(shè)計(jì)、軟件開(kāi)發(fā)、靜態(tài)網(wǎng)站、企業(yè)網(wǎng)站制作、電子商務(wù)動(dòng)態(tài)網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶(hù)投稿、用戶(hù)轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

網(wǎng)站優(yōu)化排名