python语句中对象未被定义_python 形参没有被定义???感觉遇到鬼了。。。
按照順序,給沒(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ù)名稱(chēng)來(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ù)名稱(chēng)匹配,在運(yùn)行時(shí)告訴系統(tǒng)參數(shù)name的值為‘rechar’,age的值為27
func(age = 27, name = 'rechar') ? ?#按參數(shù)名稱(chēng)匹配
func('rechar', age = 27) ? ?#name是按順序匹配,age按名稱(chēng)匹配
在python中,當(dāng)按照參數(shù)名稱(chēng)進(jìn)行匹配參數(shù)是,參數(shù)傳遞的順序是可以任意的,不要求按照函數(shù)定義中參數(shù)的順序進(jìn)行傳遞。在使用名稱(chēng)匹配時(shí),如果需要混合使用按順序匹配規(guī)則,則按順序匹配的參數(shù)必須出現(xiàn)在按key匹配的參數(shù)前,否則會(huì)報(bào)錯(cuò):
func(name = 'rechar', 27)
以上調(diào)用會(huì)報(bào)如下錯(cuò)誤:
三、函數(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
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è)”*“開(kāi)始,那么這個(gè)參數(shù)實(shí)際上是一個(gè)tuple類(lèi)型。假如傳遞的參數(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在將所有未匹配上的非按名稱(chēng)匹配的參數(shù)裝入?yún)?shù)中的tuple之后,假如還有未匹配上的按名稱(chēng)匹配的參數(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
func(name = 'rechar', age = 27, pay='1800')
TypeError: func() got an unexpected keyword argument 'pay'
它說(shuō)func這個(gè)函數(shù)沒(méi)有名稱(chēng)為”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è)以”**“開(kāi)頭的參數(shù),那么這個(gè)參數(shù)實(shí)際上是一個(gè)dict對(duì)象,它會(huì)將參數(shù)調(diào)用中所有沒(méi)有被匹配的按名稱(chēng)傳遞的參數(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)用必須按名稱(chēng)匹配
當(dāng)我們?cè)诙x函數(shù)時(shí),如果第一個(gè)參數(shù)就是”*name“參數(shù),那么可想而知,我們無(wú)法使用按順序匹配的方式傳遞,因?yàn)樗械陌错樞騻鬟f的參數(shù)值最終的歸宿都會(huì)是這里的tuple當(dāng)中。而為了給后續(xù)的參數(shù)傳遞值,我們只能使用按名稱(chēng)匹配的方法。
六、”**“參數(shù)只能出現(xiàn)在最后一個(gè)形參之后
想想為什么?其實(shí)很好理解,因?yàn)槌霈F(xiàn)在”**“形參之后的形參,無(wú)論使用按順序傳遞還是按名稱(chēng)傳遞,最終都無(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è)按名稱(chēng)傳遞的參數(shù),根據(jù)其中的key對(duì)參數(shù)進(jìn)行賦值。
總結(jié)
以上是生活随笔為你收集整理的python语句中对象未被定义_python 形参没有被定义???感觉遇到鬼了。。。的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 【译】程序员都有的这 10 个坏习惯!
- 下一篇: DirectX Repair(Direc