Python函數func定義以下:前端
def func(first, *args, second="Hello World", **kwargs): print(first) print(args) print(second) print(kwargs) func("dongfanger", "san", py="good")
運行後會輸出:python
dongfanger ('san',) Hello World {'py': 'good'}
它有四種參數:git
*args
是可變參數,arguments,存入元組。**args
是關鍵字參數,keyword arguments,存入字典。func函數的調用方式有如下這些:算法
①傳入單個定位參數。segmentfault
func("dongfanger")
dongfanger () Hello World {}
②第一個參數後的任意個參數會被*args
捕獲,存入一個元組。框架
func("dongfanger", "a", "b", "c")
dongfanger ('a', 'b', 'c') Hello World {}
③沒有明確指定名稱的關鍵字參數會被**kwargs
捕獲,存入一個字典。函數
func("dongfanger", j="1", k="2")
dongfanger () Hello World {'j': '1', 'k': '2'}
④second只能做爲關鍵字參數傳入。工具
func("dongfanger", second="cool")
dongfanger () cool {}
⑤定位函數也能做爲關鍵字參數傳入。測試
func(first="san")
san () Hello World {}
⑥字典前加上**
,其全部元素做爲單個參數傳入,同名鍵會綁定到對應具名參數上,餘下的被**args
捕獲。優化
my_dict = {"first": "dongfanger", "location": "cd", "second": "cool", "age": "secret"} func(**my_dict)
dongfanger () cool {'location': 'cd', 'age': 'secret'}
除了這四種參數,還有一種Python3新增長的僅限關鍵字參數。
僅限關鍵字參數(keyword-only argument)是Python3的新特性,func函數的second參數就是僅限關鍵字參數,「僅限」的意思是說,只能經過關鍵字參數指定,它必定不會捕獲未命名的定位參數。
假如把參數位置調整一下定義another_func函數:
def another_func(first, another_second="Hello World", *args, **kwargs): print(first) print(another_second) print(args) print(kwargs) another_func("dongfanger", "a", "b", "c")
輸出會變成:
dongfanger a # 注意這裏 ('b', 'c') {}
another_second不是僅限關鍵字參數,而只是默認值參數,由於它捕獲到了定位參數。
由此得知,定義僅限關鍵字參數,必須把它放到*args
參數後面,就像func函數同樣,反例是another_func函數。
還有第二個方法定義僅限關鍵字參數,在簽名中放一個*
:
>>> def f(a, *, b): # b是僅限關鍵字參數 ... return a, b ... >>> f(1, b=2) # 只能傳關鍵字參數 (1, 2) >>> f(1, 2) # 不能傳定位參數 Traceback (most recent call last): File "<input>", line 1, in <module> TypeError: f() takes 1 positional argument but 2 were given >>> f(1, 2, 3) # 不能傳定位參數 Traceback (most recent call last): File "<input>", line 1, in <module> TypeError: f() takes 1 positional argument but 3 were given
僅限關鍵字參數不必定要有默認值,就像b同樣,強制必須傳入實參。
函數內省的意思是說,當你拿到一個「函數對象」的時候,你能夠繼續知道,它的名字,參數定義等信息。這些信息能夠經過函數對象的屬性(一些雙下劃線的魔法方法)獲得。
對於func函數:
def func(first, *args, second="Hello World", **kwargs): print(first) print(second) print(args) print(kwargs)
和another_func函數:
def another_func(first, another_second="Hello World", *args, **kwargs): print(first) print(another_second) print(args) print(kwargs)
【__defaults__
屬性】
元組,保存着定位參數和關鍵字參數的默認值。
print(func.__defaults__) # None print(another_func.__defaults__) # ('Hello World',)
【__kwdefaults__
屬性】
字典,保存僅限關鍵字參數。
print(func.__kwdefaults__) # {'second': 'Hello World'} print(another_func.__kwdefaults__) # None
【__code__
屬性】
code對象引用,code對象自身有不少屬性,其中包括參數名稱。
print(func.__code__.co_varnames) # ('first', 'second', 'args', 'kwargs') print(another_func.__code__.co_varnames) # ('first', 'another_second', 'args', 'kwargs')
另外還能夠使用inspect庫的signature方法來查看內省中的函數參數:
from inspect import signature print(signature(func)) # (first, *args, second='Hello World', **kwargs)
框架和IDE等工具能夠使用這些信息驗證代碼。
若是刷過力扣算法題,那麼對函數註解就不會陌生。好比:
def clip(text:str, max_len:'int > 0'=80) -> str: pass
參數:
後面是註解表達式,能夠用來註解參數類型和約束。若是參數有默認值,註解放在參數名和=號之間。
能夠在函數末尾的)
和:
之間添加->
和註解表達式,來對返回值添加註解。
註解表達式能夠是任何類型,最經常使用的類型是類(如str或int)和字符串(如'int > 0'
)。
函數註解只是個註解,Python對註解所作的惟一的事情是,把它們存入函數的__annotations__
屬性中:
print(clip.__annotations__) #{'text': <class 'str'>, 'max_len': 'int > 0', 'return': <class 'str'>}
Python不作檢查,不作強制,不作驗證,什麼操做都不作!註解只是元數據,能夠供框架和IDE等工具使用。
本文介紹了Python函數的四種參數:定位參數、可變參數、默認值參數、關鍵字參數,和第五種Python3新特性參數:僅限關鍵字參數。拿到一個函數對象後,能夠經過函數屬性(一些雙下劃線的魔法方法)查看內省中的參數信息。函數註解是一種元數據,存在__annotations__
屬性中,備註函數的參數和返回值的類型,它只是個註解,Python不會作任何強制檢查。
參考資料:
《流暢的Python》
因爲我的緣由,寫文章時間變少了,雖然我仍然會努力碼字,可是更文頻率不得不下降。
從本週起公衆號改周更了!固定週五早上七點半推送!細水流長不負讀者指望!
時間跨度加長可能會致使內容連續性變弱,影視劇作法是設置上集回顧和下集預告,我借鑑了這個作法,在公衆號文章最後附加了「更文進度「,告知之前更新了什麼,之後會更新什麼。
【上文回顧】
歸檔電子書:
https://dongfanger.gitee.io/blog/
【下文預告】
順序不分前後,只列舉我想到的可能會更新的內容: