Python中*args和**kwargs的作用
寫在前面
讀代碼的過程中經(jīng)常見到這種含*args
和**kwargs
的表達(dá):
比如這個該輸出什么呢?
def foo(*args): print(args) foo(1, 2, 3, 4, 5)
這個呢?
def foo(a, *args): print('a:', a) print('args:', args) foo(1, 2, 3, 4, 5)
還有這個呢?
def bar(a,b,c): print(a,b,c) bar(*[1,2,3])
咦?∗號怎么出現(xiàn)在了一個列表前面?這樣對嗎?
*args
和**kwargs
,以及單獨(dú)的*,**到底是啥作用呢?原理是啥呢?讀完這篇文章你就徹底明白了!
*args
有兩部分構(gòu)成為——*
和args
。這里的重點(diǎn)是*
。
所以為了講清楚*args
,我們要追根溯源——理解*
的作用。
這里敲黑板,重點(diǎn)來了,這也是很多博客寫的沒有寫到的地方:∗的作用,有2個——打包參數(shù)(pack)和拆分參數(shù)(unpack)!
*argc
打包參數(shù)
例1:
def foo(*number): print(number) foo(1, 2, 3, 4, 5)
(1, 2, 3, 4, 5)
我們看到了什么?給函數(shù)5個參數(shù),成功運(yùn)行了,而且輸出是參數(shù)構(gòu)成的元組。
我們知道,如果number前不加∗號,那么很明顯foo()只能接受1個參數(shù),參數(shù)給多了少了都要報錯。而加上∗,就能成功運(yùn)行。
那么原理是什么呢?
答案是:∗把函數(shù)foo()接受到的多個參數(shù)1,2,3,4,5
,打包成了元組(1,2,3,4,5)
,賦值給了形參number。
我們可以驗(yàn)證一下:
例2:
def foo(*number): for i in number: print(i) print(type(number)) foo(1, 2, 3, 4, 5)
1
2
3
4
5
<class 'tuple'>
從例2可以看出,number確實(shí)被賦予了(1,2,3,4,5)
這個實(shí)參。
說話要講道理,詳情參見python官方文檔,這里粘個圖過來:
例3:
def foo(a, *number): print('a:', a) print('number:', number) for i in number: print(i) print(type(number)) foo(1, 2, 3, 4, 5)
a: 1 number (2, 3, 4, 5) 2 3 4 5 <class 'tuple'>
從例3可以看出,number接受到的實(shí)參變成了(2,3,4,5)
,第一個參數(shù)1
被形參a
接受走了。
所以這里我們可以給出∗作用的完整版:
∗的作用:函數(shù)接受實(shí)參時,按順序分配給函數(shù)形參,如果遇到帶∗的形參,那么就把還未分配出去的實(shí)參以元組形式打包(pack),分配給那個帶∗的形參。
可以再多幾個例子驗(yàn)證:
例4:
def foo(a, b, *number): print('a:', a) print('b:', b) print('number:', number) for i in number: print(i) print(type(number)) foo(1, 2, 3, 4, 5)
a: 1 b: 2 number: (3, 4, 5) 3 4 5 <class 'tuple'>
例5:
def foo(a, b, *number, c): print('a:', a) print('b:', b) print('c:', c) print('number:', number) for i in number: print(i) print(type(number)) foo(1, 2, 3, 4, 5)
Traceback (most recent call last): File "C:/Users/PycharmProjects/untitled10/test19.py", line 11, in <module> foo(1, 2, 3, 4, 5) TypeError: foo() missing 1 required keyword-only argument: 'c'
注意例5我特地找了個報錯的例子。自己分析一下為啥會報錯。答案是:c前面的參數(shù)帶∗,把剩下的實(shí)參都接受走了,c沒有傳入實(shí)參!
到這里,∗的打包(pack)就解釋清楚了。
還留一個小尾巴:args是啥?
答案是:args僅僅是一個約定俗成的形參的寫法,你寫成把別的也沒事,但是不利于統(tǒng)一形式。就像我們的例子里,一直用的number,也照樣運(yùn)行正確。
拆分參數(shù)
例6:
def bar(a,b,c): print(a,b,c) bar(*[1,2,3])
1 2 3
可以看出,∗這次沒有用在函數(shù)定義中,而是用在了函數(shù)調(diào)用中。在本例中的作用是啥呢?
答案是:把打包了的實(shí)參(元組或列表),拆分(unpack)成單個的,依次賦值給函數(shù)的形參。
在本例中,打包了的實(shí)參[1,2,3]
被拆分,1賦值給了形參a,2賦值給了形參b,3賦值給了形參c。
∗拆分的作用就這么簡單。理解了原理,其他的萬變不離其宗。出兩個題,練習(xí)一下:
練習(xí):以下3段程序中,哪個可以正常運(yùn)行?
例7:
def bar(a,b): print(a,b) bar(*[1, 2, 3])
例8:
def bar(a, b, c, d): print(a, b, c, d) bar(*[1, 2, 3])
例9:
def bar(a, b, c, d=10): print(a, b, c, d) bar(*[1, 2, 3])
答案是只有例9可以正常運(yùn)行。因?yàn)榘凑瘴覀冎v的原理,例7的實(shí)參3沒有對應(yīng)的形參接受,例8的形參d沒有實(shí)參賦值。
**kwargs
打包參數(shù)
上邊*args
學(xué)懂了**kwargs
也就很容易明白了。**kwargs
也有兩部分構(gòu)成為——**
和kwargs
。這里的重點(diǎn)是∗∗。沒錯,kwargs僅僅是一個約定俗成的寫法,沒有其他特殊含義,換成其他的也照用不誤,但是為了代碼可讀性,最好還是用約定俗成的。
∗∗的作用同樣也有兩個——打包參數(shù)(pack)和拆分參數(shù)(unpack)!
但是區(qū)別還是有的,簡單來說就是:
打包(pack):*args
是把多個位置參數(shù)打包成元組,**kwargs
是把多個關(guān)鍵字參數(shù)打包成字典。
拆分(unpack):*args
是把打包了的參數(shù)拆成單個的,依次賦值給函數(shù)的形參,**kwargs
是把字典的鍵值拆成單個的,依次賦值給函數(shù)的形參。
例10
def bar(**number): print(number) bar(a=1, b=2, c=3)
{'a': 1, 'b': 2, 'c': 3}
拆分參數(shù)
例11
def bar(a, b, c): print(a,b,c) bar(**{'a': 1, 'b': 2, 'c': 3})
1 2 3
注意這里有個需要注意的地方,就是用∗∗方式拆解字典給形參賦值時,需要字典的鍵名和函數(shù)形參一直,否則會報錯,自己試試就知道了。
位置參數(shù),關(guān)鍵字參數(shù),*args
,**kwargs
混用是要有一定順序的,這里我特地不寫了。因?yàn)檫@問題本身應(yīng)該不是個問題,是學(xué)習(xí)函數(shù)時應(yīng)該打下的基本功。如果真的需要,可以自己再查一下,加深印象,也比在我這伸手就拿來的好。
到此這篇關(guān)于Python中*args和**kwargs的理解的文章就介紹到這了,更多相關(guān)Python *args和**kwargs內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python中的NumPy實(shí)用函數(shù)整理之percentile詳解
這篇文章主要介紹了Python中的NumPy實(shí)用函數(shù)整理之percentile詳解,NumPy函數(shù)percentile()用于計算指定維度上數(shù)組元素的第?n?個百分位數(shù),返回值為標(biāo)量或者數(shù)組,需要的朋友可以參考下2023-09-09Python企業(yè)編碼生成系統(tǒng)總體系統(tǒng)設(shè)計概述
這篇文章主要介紹了Python企業(yè)編碼生成系統(tǒng)總體系統(tǒng)設(shè)計,簡單描述了Python企業(yè)編碼生成系統(tǒng)的功能、結(jié)構(gòu)與相關(guān)編碼實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-07-07linux centos 7.x 安裝 python3.x 替換 python2.x的過程解析
這篇文章主要介紹了linux centos 7.x 安裝 python3.x 替換 python2.x的過程解析,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-12-12Python使用循環(huán)神經(jīng)網(wǎng)絡(luò)解決文本分類問題的方法詳解
這篇文章主要介紹了Python使用循環(huán)神經(jīng)網(wǎng)絡(luò)解決文本分類問題的方法,結(jié)合實(shí)例形式詳細(xì)分析了Python神經(jīng)網(wǎng)絡(luò)相關(guān)概念、原理及解決文本分類具體操作技巧,需要的朋友可以參考下2020-01-01python 通過 pybind11 使用Eigen加速代碼的步驟
這篇文章主要介紹了python 通過 pybind11 使用Eigen加速代碼的步驟,幫助大家更好的理解和使用python,感興趣的朋友可以了解下2020-12-12Python實(shí)現(xiàn)多路視頻多窗口播放功能
這篇文章主要為大家詳細(xì)介紹了Python實(shí)現(xiàn)多路視頻多窗口播放功能的相關(guān)知識,文中的示例代碼講解詳細(xì),有需要的小伙伴可以跟隨小編一起學(xué)習(xí)一下2025-02-02