python的參數(shù)傳遞

一、前言

在python中,函數(shù)參數(shù)的定義和傳遞有以下幾種方式:

語(yǔ)法 意義
def func(name) 普通參數(shù),可以根據(jù)位置匹配,也可以根據(jù)key來(lái)匹配
def func(name=value) 默認(rèn)參數(shù),當(dāng)參數(shù)沒(méi)有傳遞時(shí),使用默認(rèn)值
def func(*iteratable) 將所有剩下的未匹配的參數(shù)收集至一個(gè)tuple中
def func(**dictionary) 將剩下未匹配的參數(shù)收集值一個(gè)dict中
def func(*, name) 必須使用key來(lái)匹配參數(shù)
def func(*other, name) 必須使用key來(lái)匹配參數(shù)
func(value) 函數(shù)調(diào)用,參數(shù)值按傳遞的參數(shù)順序匹配
func(name=value) 函數(shù)調(diào)用,參數(shù)值根據(jù)key來(lái)匹配
func(*iteratable) 函數(shù)調(diào)用,將iteratable容器中的參數(shù)展開,按位置匹配對(duì)應(yīng)的函數(shù)參數(shù)
func(**dictionary) 函數(shù)調(diào)用,將dict中的參數(shù)展開,按key值來(lái)匹配對(duì)應(yīng)的函數(shù)參數(shù)

在python中,參數(shù)可以按照順序傳遞,在調(diào)用函數(shù)時(shí),參數(shù)的值按照傳遞的順序,從左到右依次匹配。并且還可以給參數(shù)傳遞默認(rèn)值,這都很好理解,因?yàn)樵贑、C++、Java等許多語(yǔ)言中,函數(shù)的參數(shù)傳遞都是按照這種方法來(lái)傳遞的。

但python的參數(shù)定義和傳遞除了按照順序傳遞以及可以給默認(rèn)值外,它還有其它的一些特點(diǎn),在進(jìn)一步講解之前,首先說(shuō)明python中函數(shù)調(diào)用中參數(shù)匹配的順序:

  • 按照順序,給沒(méi)有key的參數(shù)賦值,意味著傳遞參數(shù)時(shí),需按順序匹配的參數(shù)必須出現(xiàn)在按key匹配的參數(shù)之前;
  • 給按照key匹配的參數(shù)賦值;
  • 將多余的按照順序匹配但未匹配的參數(shù)值歸入*name的tuple中;
  • 將多余未匹配上的按照key進(jìn)行匹配的參數(shù)值歸入**name的dict對(duì)象中;
  • 將為匹配上的且具有默認(rèn)值的參數(shù)賦默認(rèn)值

二、按key匹配參數(shù)

對(duì)于C、C++這種語(yǔ)言,在調(diào)用函數(shù)時(shí),系統(tǒng)會(huì)首先將函數(shù)地址壓入堆棧,其次按參數(shù)的從右往左的順序,一次壓入堆棧。因此,C、C++這種語(yǔ)言它們只支持按順序匹配形參。而python的做法不同,參數(shù)除了可以按順序匹配,還可以按照參數(shù)名稱來(lái)匹配。如:

def func(name, age):

    print(name, age)

對(duì)于這個(gè)函數(shù),以下的調(diào)用時(shí)等價(jià)的:

func('rechar', 27)    #按順序匹配

func(name = 'rechar', age = 27)    #按參數(shù)名稱匹配,在運(yùn)行時(shí)告訴系統(tǒng)參數(shù)name的值為‘rechar’,age的值為27

func(age = 27, name = 'rechar')    #按參數(shù)名稱匹配

func('rechar', age = 27)    #name是按順序匹配,age按名稱匹配

在python中,當(dāng)按照參數(shù)名稱進(jìn)行匹配參數(shù)是,參數(shù)傳遞的順序是可以任意的,不要求按照函數(shù)定義中參數(shù)的順序進(jìn)行傳遞。在使用名稱匹配時(shí),如果需要混合使用按順序匹配規(guī)則,則按順序匹配的參數(shù)必須出現(xiàn)在按key匹配的參數(shù)前,否則會(huì)報(bào)錯(cuò):

func(name = 'rechar', 27)

三、函數(shù)定義中的”*name“

python在給按順序匹配和按key匹配的參數(shù)賦完值后,如果發(fā)現(xiàn)調(diào)用者傳入的參數(shù)仍有未匹配上的會(huì)發(fā)生什么情況呢?看一下下面的例子:

func('rechar', 27, 32)

運(yùn)行時(shí)我們看到如下錯(cuò)誤:

Traceback (most recent call last):

  File "E:\tmp\tt.py", line 5, in <module>

    func('rechar', 27, 32)

TypeError: func() takes 2 positional arguments but 3 were given

哦,python會(huì)抱怨我們傳遞的參數(shù)太多了。那如果確實(shí)在一些情況下,我們無(wú)法保證傳遞的參數(shù)數(shù)量一定和函數(shù)需要的參數(shù)數(shù)相等怎么辦呢?這是就是iterable這種參數(shù)該登場(chǎng)的時(shí)候了,假如在定義函數(shù)定義是,我們?cè)黾恿艘粋€(gè)參數(shù),這個(gè)參數(shù)以一個(gè)”“開始,那么這個(gè)參數(shù)實(shí)際上是一個(gè)tuple類型。假如傳遞的參數(shù)比需要的多,那那些多余的參數(shù)會(huì)被放入這個(gè)tuple中。例如,

def func(name, age, *other):

    print(name, age, other)

那么,

func('rechar', 27, 32)

這個(gè)調(diào)用的輸出如下:

>>> rechar 27 (32,)

四、函數(shù)定義中的”**name“

python在將所有未匹配上的非按名稱匹配的參數(shù)裝入?yún)?shù)中的tuple之后,假如還有未匹配上的按名稱匹配的參數(shù)那情況會(huì)怎樣呢?首先來(lái)看一下下面的示例:

def func(name, age):

    print(name, age)

func(name = 'rechar', age = 27, pay='1800')

執(zhí)行時(shí),python又抱怨了:

Traceback (most recent call last):

  File "E:\tmp\tt.py", line 5, in <module>

    func(name = 'rechar', age = 27, pay='1800')

TypeError: func() got an unexpected keyword argument 'pay'

它說(shuō)func這個(gè)函數(shù)沒(méi)有名稱為”pay“的參數(shù),這種情況或許出現(xiàn)在我們函數(shù)重構(gòu)之后,原來(lái)函數(shù)時(shí)有這個(gè)參數(shù)的。而這個(gè)函數(shù)調(diào)用可能在別處沒(méi)有被修改。假設(shè)即使給了”pay“這個(gè)參數(shù),程序的正確性不受影響,沒(méi)錯(cuò),這就是”**name“參數(shù)的用武之地了。

假如在函數(shù)定義中,給函數(shù)增加一個(gè)以”**“開頭的參數(shù),那么這個(gè)參數(shù)實(shí)際上是一個(gè)dict對(duì)象,它會(huì)將參數(shù)調(diào)用中所有沒(méi)有被匹配的按名稱傳遞的參數(shù)都放入這個(gè)dict中。例如,

def func(name, age,**other):

    print(name, age, other)

func(name = 'rechar', age = 27, pay='1800')

那么運(yùn)行結(jié)果輸出,

rechar 27 {'pay': '1800'}

看到了吧,這里的other就將沒(méi)有匹配的”pay=‘1800’“收入囊中了。

五、規(guī)定調(diào)用必須按名稱匹配

當(dāng)我們?cè)诙x函數(shù)時(shí),如果第一個(gè)參數(shù)就是”*name“參數(shù),那么可想而知,我們無(wú)法使用按順序匹配的方式傳遞,因?yàn)樗械陌错樞騻鬟f的參數(shù)值最終的歸宿都會(huì)是這里的tuple當(dāng)中。而為了給后續(xù)的參數(shù)傳遞值,我們只能使用按名稱匹配的方法。

六、”**“參數(shù)只能出現(xiàn)在最后一個(gè)形參之后

想想為什么?其實(shí)很好理解,因?yàn)槌霈F(xiàn)在”“形參之后的形參,無(wú)論使用按順序傳遞還是按名稱傳遞,最終都無(wú)法到達(dá)參數(shù)值真正應(yīng)該需要到的地方。所以python規(guī)定,如果需要”“參數(shù),那它必須是最后一個(gè)形參。否則python會(huì)報(bào)語(yǔ)法錯(cuò)誤。

七、函數(shù)調(diào)用中的”*“

在表格中我們看到了有func(*iteratable)的調(diào)用,這個(gè)調(diào)用的意思是,iteratable必須是一個(gè)可迭代的容器,比如list、tuple;作為參數(shù)傳遞值,它最終傳遞到函數(shù)時(shí),不是以一個(gè)整體出現(xiàn),而是將其中的元素按照順序傳遞的方式,一次賦值給函數(shù)的形參。例如,

li = ['rechar', 27]

func(*li)

這個(gè)函數(shù)調(diào)用與

func('rechar', 27)

是等價(jià)的。

八、函數(shù)調(diào)用中的”**“

知道”“在函數(shù)調(diào)用中的效果之后,也就很好理解”*“的作用了。它是將傳遞進(jìn)來(lái)的dict對(duì)象分解,每一個(gè)元素對(duì)應(yīng)一個(gè)按名稱傳遞的參數(shù),根據(jù)其中的key對(duì)參數(shù)進(jìn)行賦值。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • http://python.jobbole.com/85231/ 關(guān)于專業(yè)技能寫完項(xiàng)目接著寫寫一名3年工作經(jīng)驗(yàn)的J...
    燕京博士閱讀 7,629評(píng)論 1 118
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,969評(píng)論 19 139
  • 一、python 變量和數(shù)據(jù)類型 1.整數(shù) Python可以處理任意大小的整數(shù),當(dāng)然包括負(fù)整數(shù),在Python程序...
    績(jī)重KF閱讀 1,783評(píng)論 0 1
  • 最近在慕課網(wǎng)學(xué)習(xí)廖雪峰老師的Python進(jìn)階課程,做筆記總結(jié)一下重點(diǎn)。 基本變量及其類型 變量 在Python中,...
    victorsungo閱讀 1,740評(píng)論 0 5
  • Python的參數(shù)傳遞概述 根據(jù)Python的官方文檔,Python中的參數(shù)傳遞機(jī)制是所謂的pass by ass...
    耀凱考前突擊大師閱讀 484評(píng)論 0 1