成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Python中函數參數傳遞方法*args, **kwargs,還有其他

開發 前端
本文將討論Python的函數參數。我們將了解*args和**kwargs,/和*的都是什么,雖然這個問題是一個基本的python問題,但是在我們寫代碼時會經常遇到,比如timm中就大量使用了這樣的參數傳遞方式。

本文將討論Python的函數參數。我們將了解args和**kwargs,/和的都是什么,雖然這個問題是一個基本的python問題,但是在我們寫代碼時會經常遇到,比如timm中就大量使用了這樣的參數傳遞方式。

定義和傳遞參數

parameters 和arguments 之間的區別是什么?

許多人交替使用這些術語,但它們是有區別的:

  • Parameters 是函數定義中定義的名稱
  • Arguments是傳遞給函數的值

紅色的是parameters , 綠色的是arguments。

傳遞參數的兩種方式

我們可以按位置和關鍵字傳遞參數。在下面的例子中,我們將值hello作為位置參數傳遞。值world 用關鍵字傳遞的。

 def the_func(greeting, thing):
print(greeting + ' ' + thing)

the_func('hello', thing='world')

位置參數和kwargs(關鍵字參數)之間的區別在于傳遞位置參數的順序很重要。如果調用the_func('world', 'hello')它會打印world hello。傳遞kwargs的順序并不重要:

the_func('hello', 'world')                  # -> 'hello world'
the_func('world', 'hello') # -> 'world hello'
the_func(greeting='hello', thing='world') # -> 'hello world'
the_func(thing='world', greeting='hello') # -> 'hello world'
the_func('hello', thing='world') # -> 'hello world'

只要kwarg在位置參數之后,就可以混合和匹配位置參數和關鍵字參數,以上就是我們在python教程中經常看到的內容,下面我們繼續。

函數參數

我們將演示6個函數參數傳遞的方法,這些方法能夠覆蓋到所有的問題。

1、如何獲得所有未捕獲的位置參數

使用*args,讓它接收一個不指定數量的形參。

def multiply(a, b, args):
result = a * b
for arg in args:
result = result * arg
return result

在這個函數中,我們通常定義前兩個參數(a和b)。然后使用args將所有剩余參數打包到一個元組中。可以把*看作是獲取到了其他沒有處理的參數,并將它們收集到一個名為“args”的元組變量中:

multiply(1, 2)          # returns 2
multiply(1, 2, 3, 4) # returns 24

最后一次調用將值1賦給參數a,將2賦給參數b,并將arg變量填充為(3,4)。由于這是一個元組,我們可以在函數中循環它并使用這些值進行乘法!

2、如何獲得所有未捕獲的關鍵字參數

與*args類似,這次是兩個星號**kwargs

def introduce(firstname, lastname, **kwargs):
introduction = f"I am {firstname} {lastname}"
for key, value in kwargs.items():
introduction += f" my {key} is {value} "
return introduction

**kwargs關鍵字會將所有不匹配的關鍵字參數存儲在一個名為kwargs的字典中。然后可以像上面的函數一樣訪問這個字典。

 print(introduce(firstname='mike', lastname='huls'))
# returns "I am mike huls"

print(introduce(firstname='mike', lastname='huls', age=33, website='mikehuls.com'))
# I am mike huls my age is 33 my website is overfit.cn

3、如果想只接受關鍵字參數,那怎么設計

可以強制函數只接受關鍵字參數。

 def transfer_money(*, from_account:str, to_account:str, amount:int):
print(f'Transfering ${amount} FORM {from_account} to {to_account}')

transfer_money(from_account='1234', to_account='6578', amount=9999)
# won't work: TypeError: transfer_money() takes 0 positional arguments but 1 positional argument (and 2 keyword-only arguments) were given
transfer_money('1234', to_account='6578', amount=9999)
# won't work: TypeError: transfer_money() takes 0 positional arguments but 3 were given
transfer_money('1234', '6578', 9999)

在上面的函數中,*星號獲得了了所有不匹配的位置參數,但是并沒有一個變量來接受它,也就是被忽略了。

4、如何設計函數只接受位置參數

下面是一個只允許位置參數的函數示例:

 def the_func(arg1:str, arg2:str, /):
print(f'provided {arg1=}, {arg2=}')

# These work:
the_func('num1', 'num2')
the_func('num2', 'num1')

# won't work: TypeError: the_func() got some positional-only arguments passed as keyword arguments: 'arg1, arg2'
the_func(arg1='num1', arg2='num2')
# won't work: TypeError: the_func() got some positional-only arguments passed as keyword arguments: 'arg2'
the_func('num1', arg2='num2')

函數定義中的/強制在它之前的所有參數都是位置參數。這并不意味著/后面的所有參數都必須是kwarg-only;這些可以是位置和關鍵字。

看到這個你肯定會想,為什么想要這個?這不會降低代碼的可讀性嗎?,我也覺得你說的非常正確,當定義一個非常明確的函數時,不需要關鍵字參數來指定它的功能。例如:

def exceeds_100_bytes(x, /) -> bool:
return x.__sizeof__() > 100

exceeds_100_bytes('a')
exceeds_100_bytes({'a'})

在這個例子中,正在檢查'a'的內存大小是否超過100字節。因為這個x對于我們來說他的名字不重要,在調用函數的時候不需要指定x= ' a '。比如說我們最常用的len,如果你調用len(__obj=[]) 這樣看起來是不是有點呆萌,因為len是這么定義的def len(__obj: Sized) -> int:

5、混合和匹配

作為一個例子,我們將看看前面討論過的len函數。這個函數只允許位置參數。我們將通過允許開發人員選擇是否計算重復項來擴展此函數,比如用kwargs傳遞這個關鍵字:

 def len_new(x, /, *, no_duplicates=False):
if (no_duplicates):
return len(list(set([a for a in x])))
return len(x)

想計算變量x的len,只能按位置傳遞x形參的參數,因為它前面有一個/。no_duplicate參數必須與關鍵字一起傳遞,因為它跟在后面。讓我們看看這個函數都可以怎么調用:

print(len_new('aabbcc'))                                  # returns 6
print(len_new('aabbcc', no_duplicates=True)) # returns 3
print(len_new([1, 1, 2, 2, 3, 3], no_duplicates=False)) # returns 6
print(len_new([1, 1, 2, 2, 3, 3], no_duplicates=True)) # returns 3

# Won't work: TypeError: len_() got some positional-only arguments passed as keyword arguments: 'x'
print(len_new(x=[1, 1, 2, 2, 3, 3]))
# Won't work: TypeError: len_new() takes 1 positional argument but 2 were given
print(len_new([1, 1, 2, 2, 3, 3], True))

6、最后把它們合在一起

下面的函數是一個非常極端的例子,說明了如何組合前面討論的所有技術:它強制前兩個參數以位置方式傳遞,接下來的兩個參數可以以位置方式傳遞,并且帶有關鍵字,然后是兩個只有關鍵字的參數,然后我們用**kwargs捕獲剩下的未捕獲的參數。

 def the_func(pos_only1, pos_only2, /, pos_or_kw1, pos_or_kw2, *, kw1, kw2, **extra_kw):
# cannot be passed kwarg <-- | --> can be passed 2 ways | --> can only be passed by kwarg
print(f"{pos_only1=}, {pos_only2=}, {pos_or_kw1=}, {pos_or_kw2=}, {kw1=}, {kw2=}, {extra_kw=}")

調用方式如下:

# works (pos_or_kw1 & pow_or_k2 can be passed positionally and by kwarg)
pos_only1='pos1', pos_only2='pos2', pos_or_kw1='pk1', pos_or_kw2='pk2', kw1='kw1', kw2='kw2', extra_kw={}
pos_only1='pos1', pos_only2='pos2', pos_or_kw1='pk1', pos_or_kw2='pk2', kw1='kw1', kw2='kw2', extra_kw={}
pos_only1='pos1', pos_only2='pos2', pos_or_kw1='pk1', pos_or_kw2='pk2', kw1='kw1', kw2='kw2', extra_kw={'kw_extra1': 'extra_kw1'}

# doesnt work, (pos1 and pos2 cannot be passed with kwarg)
# the_func(pos_only1='pos1', pos_only2='pos2', pos_or_kw1='pk1', pos_or_kw2='pk2', kw1='kw1', kw2='kw2')

# doesnt work, (kw1 and kw2 cannot be passed positionally)
# the_func('pos1', 'pos2', 'pk1', 'pk2', 'kw1', 'kw2')

總結

看著很亂是吧,這就對了。因為python在設計時是一個很寬松的語言,并沒有那么多的規范,用的人越多使用方法就越多,就變成了這樣。

那么回到第一張圖:

def func(x,/,y,,z,**k):

(x,/,y,,z,**k):是函數的參數。總共有四個參數:

  • x: 是一個常規參數,這意味著它可以按位置傳遞,也可以按關鍵字傳遞。
  • /,: 是一個參數分隔符,將僅限位置的參數與其他參數分開。與前面的x結合,意味著x只能按位置傳遞。
  • y: 時另一個常規參數。
  • *: 是一個參數分隔符,用于分隔僅限位置參數和僅限關鍵字參數。它意味著后面的z只能通過關鍵字傳遞。
  • z: 是一個僅限關鍵字的參數。
  • **k: 這是一個參數,將所有剩余的關鍵字參數收集到一個名為' k '的字典中。

這樣解釋是不是就很明白了。

我們今天介紹的這個例子雖然在看源代碼時沒有遇到這么復雜的情況,但是在 面試 的時候還真有人問(雖然我覺得沒啥用),所以最好還是知道一些,以免尷尬。

如果你忘記了,這里可以教你一個變通的辦法,可以使用類似的回答:

上面的參數傳遞在開發時并不常用,因為對于開發規范來說,應該保證代碼的可讀性,我們這邊遵循的開發規范是:

1、盡量不要在函數定義中將可變位置參數 *args 和可變關鍵字參數 **kwargs 放在一起,因為這樣會讓函數的調用方式變得不太直觀。

2、在使用可變參數時,要保證函數的行為是可預測的。上面函數中的進行了太多的python語法糖,對于理解該函數的參數會造成很大的困惑,也就是可讀性太差,我們在進行codereview(如果你了解什么是codereview就說,不了解就說組長檢查)/組長merge代碼 時會直接要求返工,所以我們在實際開發時是不會用這個的。

對于我閱讀的開源代碼,也都基本上使用的是 **kwargs 這種情況(這里可以舉兩個例子),還沒有看到有人寫這么亂的代碼,我想要是寫這樣的代碼估計開源的人也會被人吐糟(這里自己可以自行延伸),所以這些參數傳遞的規則我在學習的時候看到過,但是實際中沒見過真正使用,就不太記住了。

回到本文,我們介紹了設計函數參數的所有方法,并了解了如何混合和匹配它們,雖然后面幾個內容可能你一輩子也不會用到,但是了解一下也是好的,因為萬一呢。

責任編輯:華軒 來源: DeepHub IMBA
相關推薦

2025-02-12 10:51:51

2010-03-11 11:07:37

Python函數參數

2024-12-18 08:01:31

2023-11-17 14:10:08

C++函數

2009-12-17 17:04:09

Ruby函數參數傳遞

2025-01-17 10:52:26

定義函數編程Python

2009-11-26 09:18:56

PHP函數參數傳遞方法

2010-07-26 13:13:33

Perl函數參數

2024-04-28 11:36:07

LambdaPython函數

2012-02-21 14:04:15

Java

2024-10-06 14:01:47

Python裝飾器對象編程

2021-06-21 09:30:12

@wraps 修飾器Python

2009-09-04 16:10:49

JSP頁面間傳遞參數

2009-08-27 17:47:18

C#匿名方法作為參數傳

2025-06-05 02:43:00

2022-11-06 21:50:59

Python編程函數定義

2025-04-02 12:00:00

開發日志記錄Python

2010-01-25 09:57:39

C++函數參數

2024-03-07 12:40:28

Python*args開發

2009-09-07 03:23:40

C# Main方法
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲一区二区免费视频 | 欧美在线亚洲 | 亚洲高清在线观看 | 欧美成人猛片aaaaaaa | 色屁屁在线观看 | 在线亚洲免费视频 | 亚洲精品一 | www久久99| 久久aⅴ乱码一区二区三区 亚洲国产成人精品久久久国产成人一区 | 国产激情91久久精品导航 | 一级黄色夫妻生活 | 日本久久久影视 | 成人久久18免费 | 日本一区二区在线视频 | 中文字幕亚洲一区 | 国产精品欧美一区二区三区不卡 | 福利视频二区 | 亚洲欧洲日韩 | 91精品国产91久久久久久最新 | 九九久久99 | 久久一二区 | 欧美激情综合色综合啪啪五月 | 日本一区二区高清不卡 | 国产国产精品久久久久 | 亚洲xx在线 | 国产免费拔擦拔擦8x高清 | 亚洲视频中文字幕 | 一区二区不卡高清 | 久久毛片| 成人欧美一区二区三区黑人孕妇 | 秋霞影院一区二区 | 爱高潮www亚洲精品 中文字幕免费视频 | 久久久成人免费视频 | 欧美一区二区三区在线播放 | 国产乱码精品一区二区三区五月婷 | 亚洲精品一二三区 | 久久久国产精品视频 | 久久久www成人免费无遮挡大片 | a爱视频| 久久毛片 | 欧美高清一区 |