基本概念
讓我們首先來復(fù)習(xí)一下函數(shù)的基本定義。
在Python中,函數(shù)是一段可重用的代碼塊,用于執(zhí)行特定的任務(wù)。我們可以使用 def 關(guān)鍵字來定義函數(shù),例如:
def greet(name):
print("Hello, " + name)
greet("Tom") # 輸出:Hello, Tom
greet("Jane") # 輸出:Hello, Jane
其中,函數(shù)的定義包括函數(shù)名、參數(shù)列表和函數(shù)體。
- 函數(shù)名是函數(shù)的唯一標(biāo)識(shí),由開發(fā)人員來定義。
- 參數(shù)列表是函數(shù)所需的輸入,可以是一個(gè)或多個(gè)參數(shù)。
- 函數(shù)體是函數(shù)所要執(zhí)行的代碼,可以包括多行語句。
我們還可以使用 return 關(guān)鍵字來向調(diào)用函數(shù)的代碼返回值:
def add(x, y):
return x + y
result = add(3, 4)
print(result) # 輸出:7
在復(fù)習(xí)了函數(shù)的基本概念之后,我們將在本節(jié)詳細(xì)講解函數(shù)中參數(shù)的用法。
基本用法
首先,在Python中,函數(shù)的參數(shù)是用來接受輸入的值的。函數(shù)可以接受一個(gè)或多個(gè)參數(shù),并且參數(shù)可以是任意類型的值。
當(dāng)我們?cè)谑褂?def 關(guān)鍵字來定義函數(shù)時(shí),可以同時(shí)指定參數(shù)列表:
def greet(name, greeting):
print(greeting + ", " + name)
greet("Tom", "Hello") # 輸出:Hello, Tom
greet("Jane", "Hi") # 輸出:Hi, Jane
上面的代碼定義了一個(gè)名為 greet 的函數(shù),接受兩個(gè)參數(shù):name 和 greeting。在函數(shù)調(diào)用時(shí),使用者需要提供與參數(shù)列表中所定義的參數(shù)數(shù)量相同的值。
從不同的角度看,我們可以將Python中的函數(shù)參數(shù)類型分為以下幾種:
- 位置參數(shù)
- 關(guān)鍵字參數(shù)
- 默認(rèn)參數(shù)
- 可變參數(shù)
- 限定關(guān)鍵字參數(shù)
- 限定位置參數(shù)
- 參數(shù)組合
下面,就讓我們來一一講解。
位置參數(shù)
在Python中,位置參數(shù)(Positional Parameter)是指函數(shù)所接受的固定數(shù)量的參數(shù),按照它們?cè)趨?shù)列表中的順序來解析。
一般情況下,我們?cè)谑褂?def 關(guān)鍵字來定義函數(shù)時(shí),給出的就是位置參數(shù)列表,如同前面的例子所演示的:
def greet(name, greeting):
print(greeting + ", " + name)
greet("Tom", "Hello") # 輸出:Hello, Tom
greet("Jane", "Hi") # 輸出:Hi, Jane
上面的代碼定義了一個(gè)名為 greet 的函數(shù),接受的兩個(gè)參數(shù)就是位置參數(shù):name 和 greeting。
在函數(shù)調(diào)用時(shí),使用者需要提供與參數(shù)列表中所定義的參數(shù)數(shù)量相同的值,并按照它們?cè)趨?shù)列表中的順序來解析。由于使用者需要嚴(yán)格按照參數(shù)列表定義的順序提供對(duì)應(yīng)的參數(shù),所以,我們就把這種參數(shù)類型就叫做位置參數(shù)。
位置參數(shù)是Python中最常用的參數(shù)類型,因?yàn)樗芎?jiǎn)單而有效地傳遞輸入。
關(guān)鍵字參數(shù)
在使用位置參數(shù)時(shí),我們只需要按照位置順序提供參數(shù)即可;但如果一個(gè)函數(shù)的參數(shù)較多,使用者就可能很難記住所有的參數(shù)順序,而在使用的時(shí)候出現(xiàn)錯(cuò)誤。例如,在上面的例子中,使用者可能會(huì)忘記到底是需要把“Tom”放在“Hello”的前面,還是把“Hello”放在“Tom”的前面。
為了避免這種情況,Python引入了關(guān)鍵字參數(shù)類型。
在Python中,關(guān)鍵字參數(shù)(Keyword Parameter)是指函數(shù)調(diào)用時(shí),參數(shù)名和參數(shù)值都是顯式指定的參數(shù)。
我們可以通過在調(diào)用函數(shù)時(shí)指定參數(shù)名和參數(shù)值的方式,來使用關(guān)鍵字參數(shù):
def greet(name, greeting):
print(greeting + ", " + name)
greet(name="Tom", greeting="Hello") # 輸出:Hello, Tom
greet(greeting="Hi", name="Jane") # 輸出:Hi, Jane
此例中的函數(shù)與前面示例中的函數(shù)是一樣的,但是我們?cè)谡{(diào)用它時(shí),顯式地指定了參數(shù)名和參數(shù)值。這樣,我們的調(diào)用函數(shù)看起來就非常直接和明了,不會(huì)出現(xiàn)忘記參數(shù)順序的問題。
另外,關(guān)鍵字參數(shù)還有一個(gè)好處,就是使用者可以忽略參數(shù)列表的實(shí)際順序。在函數(shù)調(diào)用時(shí),使用者可以以任意順序提供參數(shù)名和參數(shù)值。在上例中,greet(greeting="Hi", name="Jane") 與 greet(name="Jane", greeting="Hi") 是等價(jià)的。
總之,相比于位置參數(shù),關(guān)鍵字參數(shù)可以給予調(diào)用者更好的可讀性和更好的調(diào)用體驗(yàn)。
默認(rèn)參數(shù)
在上面的示例中,我們?cè)谡{(diào)用函數(shù) greet() 的時(shí)候,每次都需要提供 greeting 參數(shù),用來顯示歡迎語。假設(shè)我們的歡迎語始終都是“Hello”,那么每次調(diào)用都需要提供“Hello”就會(huì)顯得很繁瑣。在這種情況下,我們就需要使用默認(rèn)參數(shù)了。
在Python中,默認(rèn)參數(shù)(Default Parameter)是指在定義函數(shù)時(shí)為參數(shù)指定默認(rèn)值的參數(shù)。
我們可以使用 def 關(guān)鍵字來定義函數(shù),并指定默認(rèn)參數(shù)值:
def greet(name, greeting="Hello"):
print(greeting + ", " + name)
greet("Tom") # 輸出:Hello, Tom
greet("Tom", "Hi") # 輸出:Hi, Tom
在上面的代碼中,我們?yōu)楹瘮?shù) greet() 的參數(shù) greeting 指定了一個(gè)默認(rèn)值:"Hello"。這樣,如果使用者不提供 greeting 參數(shù),函數(shù)就使用默認(rèn)的參數(shù)值 “Hello” 來代替;而如果使用者提供了參數(shù)值,那么函數(shù)就會(huì)使用提供的參數(shù)值。
在實(shí)際開發(fā)中,默認(rèn)參數(shù)非常有用,因此它能簡(jiǎn)化函數(shù)的定義,并且在調(diào)用函數(shù)時(shí)給予使用者更好的體驗(yàn)。
可變參數(shù)
在前面的示例中,我們?cè)诙x函數(shù)時(shí)指定的參數(shù)都是固定的。例如,greet() 函數(shù)有且僅有兩個(gè)參數(shù),我們可以按照下面的方式調(diào)用 greet("Tom", "Hello"),但不能使用這樣的方式 greet("Tom", "Jerry", "Hello"),雖然,我們希望同時(shí)歡迎多個(gè)人員。
為了解決這樣的問題,Python提供了可變長(zhǎng)度參數(shù)。
在Python中,可變參數(shù)(Variable-length Parameter)是指函數(shù)能接受任意數(shù)量的參數(shù)的參數(shù)。
Python提供了兩種特殊的運(yùn)算符來定義可變參數(shù): * 和 ** 。在實(shí)際開發(fā)中,我們通常使用 *args 和 **kwargs 來定義。這不是Python的命名規(guī)則,只是約定俗成的使用慣例。
-
*args:允許函數(shù)接受可變數(shù)量的位置參數(shù)。傳遞給*args的參數(shù)會(huì)被收集到一個(gè)元組中。def my_function(*args): # 使用傳遞給 *args 的參數(shù)執(zhí)行函數(shù)的代碼 my_function(1, 2, 3, 4) # args = (1, 2, 3, 4) -
**kwargs:允許函數(shù)接受可變數(shù)量的關(guān)鍵字參數(shù)。傳遞給**kwargs的參數(shù)會(huì)被收集到一個(gè)字典中。def my_function(**kwargs): # 使用傳遞給 **kwargs 的參數(shù)執(zhí)行函數(shù)的代碼 my_function(a=1, b=2, c=3) # kwargs = {'a': 1, 'b': 2, 'c': 3}
在Python中使用可變參數(shù),能夠?yàn)槲覀兲峁┮韵碌暮锰帲?/p>
-
允許函數(shù)接受任意數(shù)量的位置參數(shù)
如前所述,在Python中使用可變參數(shù)的常見用例之一是當(dāng)我們想要定義一個(gè)可以接受任意數(shù)量位置參數(shù)的函數(shù)的時(shí)候。例如,假如我們需要一個(gè)函數(shù)能夠允許多個(gè)文件名作為輸入,我們就可以使用
*args語法來定義此類函數(shù):def print_files(*file_names): for file_name in file_names: with open(file_name) as f: print(f.read()) print_files("file1.txt", "file2.txt", "file3.txt")在此示例中,
print_files()函數(shù)使用*file_names語法接受可變數(shù)量的位置參數(shù)。函數(shù)然后逐個(gè)打開每個(gè)文件并打印其內(nèi)容。通過使用可變參數(shù),我們可以輕松地將任何數(shù)量的文件名傳遞給函數(shù),而不必手動(dòng)逐個(gè)指定它們。 -
允許函數(shù)接受任意數(shù)量的關(guān)鍵字參數(shù)
在Python中使用可變參數(shù)的另一個(gè)用例是當(dāng)我們需要一個(gè)函數(shù)能夠接受任意數(shù)量的關(guān)鍵字參數(shù)的時(shí)候。例如,我們可能有一個(gè)函數(shù)需要將可變數(shù)量的配置選項(xiàng)作為輸入,這時(shí)我們就可以使用
**kwargs語法來定義此類函數(shù):def configure(**options): for key, value in options. Items(): print(f"Setting {key} to {value}") configure(database="mydb", username="myuser", password="mypassword")在此例中,
configure()函數(shù)使用**options語法接受可變數(shù)量的關(guān)鍵字參數(shù)。函數(shù)然后打印傳入的每個(gè)鍵值對(duì)。通過使用可變參數(shù),我們可以輕松地將任意數(shù)量的配置選項(xiàng)傳遞給函數(shù),而不必為每個(gè)選項(xiàng)定義單獨(dú)的參數(shù)。 -
允許函數(shù)同時(shí)組合位置和關(guān)鍵字參數(shù)
如果我們需要一個(gè)函數(shù)可以同時(shí)可變數(shù)量的接受位置和可變數(shù)量的關(guān)鍵字參數(shù),我們就可以同時(shí)使用
*args語法和**kwargs語法。def process_inputs(*args, **kwargs): # 處理任何位置參數(shù) for arg in args: # 對(duì) arg 進(jìn)行某些處理 pass # 處理任何關(guān)鍵字參數(shù) for key, value in kwargs.items(): # 對(duì) key 和 value 進(jìn)行某些處理 pass process_inputs(1, 2, 3, database="mydb", username="myuser", password="mypassword")在此例中,
process_inputs()函數(shù)使用*args語法接受任意數(shù)量的位置參數(shù)和**kwargs語法接受任意數(shù)量的關(guān)鍵字參數(shù)。函數(shù)然后逐個(gè)處理每個(gè)參數(shù),首先處理任何位置參數(shù),然后處理任何關(guān)鍵字參數(shù)。通過這樣的位置和關(guān)鍵字參數(shù)的組合,我們可以創(chuàng)建一個(gè)非常靈活的函數(shù),可以接受各種類型的輸入。
總之,在Python中使用可變參數(shù)可以讓我們能夠編寫更靈活、能夠接受更廣泛類型輸入的函數(shù)。通過有效地使用 *args 和 **kwargs,我們可以創(chuàng)建非常強(qiáng)大、而又易于使用的函數(shù)。
限定關(guān)鍵字參數(shù)
我們?cè)谇懊鎸W(xué)習(xí)關(guān)鍵字參數(shù)時(shí),可以看到,其本質(zhì)上是位置參數(shù)的變體;也就是說,我們可以不適用關(guān)鍵字參數(shù)來傳遞參數(shù)值,而只是簡(jiǎn)單地使用位置參數(shù)。但是,如果我們需要指定函數(shù)調(diào)用者必須使用關(guān)鍵字參數(shù)來傳遞參數(shù)值時(shí),該怎么辦呢?此時(shí),就需要限定關(guān)鍵字參數(shù)出場(chǎng)了。
在Python中,限定關(guān)鍵字參數(shù)(keyword-only parameter)是指在函數(shù)定義時(shí),參數(shù)名和參數(shù)值必須使用關(guān)鍵字參數(shù)形式指定的參數(shù)。很多時(shí)候,限定關(guān)鍵字參數(shù)也被稱為“命名關(guān)鍵字參數(shù)”。
我們可以在函數(shù)定義時(shí)使用 * 來聲明限定關(guān)鍵字參數(shù)列表:
def greet(*, name, greeting):
print(greeting + ", " + name)
greet(name="Tom", greeting="Hello") # 輸出:Hello, Tom
greet(greeting="Hi", name="Jane") # 輸出:Hi, Jane
上面的代碼定義了一個(gè)名為 greet() 的函數(shù),接受兩個(gè)命名關(guān)鍵字參數(shù):name 和 greeting。
由于在這兩個(gè)參數(shù)前面使用了 * 來聲明命名關(guān)鍵字參數(shù),因此,name 和 greeting 皆為命名關(guān)鍵字。也就是說,在函數(shù)調(diào)用時(shí),調(diào)用者必須使用關(guān)鍵字參數(shù)的形式來傳遞參數(shù)值。
不過,命名關(guān)鍵字參數(shù)是Python中一種很少使用的參數(shù)類型,它主要用于強(qiáng)制使用關(guān)鍵字參數(shù)的形式調(diào)用函數(shù),以提高可讀性和調(diào)用體驗(yàn)。
限定位置參數(shù)
在Python 3.8之前,函數(shù)定義中的所有參數(shù)都被視為“positional-or-keyword”,也就是“既可以是-位置-也可以是-關(guān)鍵字”參數(shù);這就意味著它們可以作為位置參數(shù)傳遞或作為關(guān)鍵字參數(shù)傳遞。不過,Python 3.8引入了一個(gè)新的概念,允許開發(fā)人員使用 / 分隔符來定義限定位置參數(shù)。
在Python中,限定位置參數(shù)(Positional-only Parameter)是只能作為位置參數(shù)傳遞、而不能作為關(guān)鍵字參數(shù)傳遞的參數(shù)。如果需要在函數(shù)中定義限定位置參數(shù)參數(shù),我們只需在參數(shù)列表中將參數(shù)名放在 / 分隔符之前即可。
以下是一個(gè)示例:
def my_func(pos1, pos2, /, pos3):
print(f"pos_only1: {pos1}")
print(f"pos_only2: {pos2}")
print(f"pos3: {pos3}")
在這個(gè)示例中,/ 分隔符出現(xiàn)在前兩個(gè)參數(shù) pos1 和 pos2 之后,表明這兩個(gè)參數(shù)必須作為位置參數(shù)、按順序傳遞;而第三個(gè)參數(shù) pos3 出現(xiàn)在 / 分隔符之后,表示它既可以作為位置參數(shù)傳遞,也可以作為關(guān)鍵字參數(shù)傳遞。
以下是一個(gè)調(diào)用該函數(shù)的示例:
my_func(1, 2, pos3=3)
輸出的內(nèi)容如下:
pos_only1: 1
pos_only2: 2
pos3: 3
在此例中,我們將值 1 和 2 作為位置參數(shù)傳遞,而將 3 作為關(guān)鍵字參數(shù)傳遞。由于 pos1 和 pos2 是限定位置參數(shù),因此我們不能將它們作為關(guān)鍵字參數(shù)傳遞。
參數(shù)組合
在Python中,我們可以使用上述的多種參數(shù)類型來定義函數(shù)。
例如,我們可以定義一個(gè)函數(shù),同時(shí)使用位置參數(shù)、默認(rèn)參數(shù)和關(guān)鍵字參數(shù):
def greet(greeting, name="world", punctuation="!"):
print(greeting + ", " + name + punctuation)
greet("Hello") # 輸出:Hello, world!
greet("Hi", "Tom") # 輸出:Hi, Tom!
greet("Welcome", punctuation=".") # 輸出:Welcome, world.
上面的代碼定義了一個(gè)名為 greet() 的函數(shù),接受一個(gè)位置參數(shù)和兩個(gè)可選參數(shù)。在函數(shù)調(diào)用時(shí),調(diào)用者可以根據(jù)需要提供參數(shù)值,或者使用默認(rèn)參數(shù)值。
再看一個(gè)例子,我們可以定義一個(gè)函數(shù),同時(shí)使用位置參數(shù)、默認(rèn)參數(shù)、可變參數(shù)、關(guān)鍵字參數(shù)、限定關(guān)鍵字參數(shù)和限定位置參數(shù):
def example_function(pos1, pos2, /, kwonly1=None, *, kwonly2=False, *args, **kwargs):
print(f"pos1: {pos1}")
print(f"pos2: {pos2}")
print(f"kwonly1: {kwonly1}")
print(f"kwonly2: {kwonly2}")
print(f"args: {args}")
print(f"kwargs: {kwargs}")
在這個(gè)函數(shù)中,我們有:
- 兩個(gè)位置參數(shù)(
pos1和pos2)在/分隔符之前定義。它們必須按順序在調(diào)用函數(shù)時(shí)提供。 - 一個(gè)限定關(guān)鍵字參數(shù)(
kwonly1),默認(rèn)值為None。此參數(shù)可以在調(diào)用函數(shù)時(shí)省略,如果未提供,則默認(rèn)為None。 - 一個(gè)限定關(guān)鍵字參數(shù)(
kwonly2),默認(rèn)值為False。此參數(shù)也可以在調(diào)用函數(shù)時(shí)省略,如果未提供,則默認(rèn)為False。 - 一個(gè)可變參數(shù)(
*args)在/分隔符之后定義。該參數(shù)可以接受任意數(shù)量的額外位置參數(shù),并將它們收集到一個(gè)元組中。 - 一個(gè)可變長(zhǎng)度關(guān)鍵字參數(shù)(
**kwargs)在*args參數(shù)之后定義。該參數(shù)可以接受任意數(shù)量的額外關(guān)鍵字參數(shù),并將它們收集到一個(gè)字典中。
以下是一個(gè)調(diào)用此函數(shù)的示例,其中使用了不同類型的參數(shù):
example_function(1, 2, 3, kwonly1="hello", kwonly2=True, 4, 5, x=6, y=7, z=8)
輸出的內(nèi)容如下所示:
pos1: 1
pos2: 2
kwonly1: hello
kwonly2: True
args: (3, 4, 5)
kwargs: {'x': 6, 'y': 7, 'z': 8}
在此例中,我們按照正確的順序提供了必需的限定位置參數(shù) pos1 和 pos2;我們還為兩個(gè)限定關(guān)鍵字參數(shù) ( kwonly1 和 kwonly2)傳遞了參數(shù)值;我們包括了一些額外的位置參數(shù)( 4 和 5 ),這些參數(shù)都被收集到了 args 元組中,以及一些額外的關(guān)鍵字參數(shù)( x=6、y=7 和 z=8 ),這些參數(shù)都被收集到了 kwargs 字典中。
總結(jié)
綜上所述,在Python中,函數(shù)是通過參數(shù)接受輸入的值的,函數(shù)可以接受一個(gè)或多個(gè)參數(shù),并且參數(shù)可以是任意類型的值。
在實(shí)際使用中,我們可以將Python中的函數(shù)參數(shù)類型分為以下幾種:
- 位置參數(shù)
- 關(guān)鍵字參數(shù)
- 默認(rèn)參數(shù)
- 可變參數(shù)
- 限定關(guān)鍵字參數(shù)
- 限定位置參數(shù)
- 參數(shù)組合
通過熟練地使用不同類型的參數(shù),我們可以編寫出非常靈活的函數(shù),從而讓我們的代碼更易于重用和維護(hù)。
學(xué)習(xí)Python基礎(chǔ),請(qǐng)瀏覽《百例講透Python基礎(chǔ)》。