今天跟大家介紹一下 python 當(dāng)中星號(*
)的一些用法
首先大家最常見的就是在 python 中 *
是乘法運(yùn)算符,實(shí)現(xiàn)乘法
sum = 5 * 5 # 25
除此之外,還有一種常見的用法就是 *
號操作符在函數(shù)中的用法
- 單星號(
*
)在函數(shù)中用法
舉個(gè)例子,你有一個(gè)函數(shù),用來實(shí)現(xiàn)兩個(gè)數(shù)求和
def add(num1, num2):
return num1 + num2
print(add(1,2)) # 3
上面這個(gè)函數(shù)只能接收固定個(gè)數(shù)(兩個(gè))的參數(shù),那如果我想要求任意數(shù)量的值的和該怎么辦?
可以在形參部分加一個(gè) *
號
def add(*num):
sum = 0
for i in num:
sum += i
return sum
在函數(shù)定義的時(shí)候形參前面加一個(gè) *
號,就可以用任意數(shù)量的參數(shù)來調(diào)用函數(shù),如果檢查數(shù)據(jù)類型,會(huì)發(fā)現(xiàn) num 是一個(gè)元組(tuple)
即將所有傳入函數(shù)的位置參數(shù)打包成一個(gè)元組
#使用任意數(shù)量的參數(shù)來調(diào)用函數(shù)
add(1, 2, 3, 4) # 10
那如果函數(shù)在定義的時(shí)候就已經(jīng)固定了形參的個(gè)數(shù)
def add(num1, num2, num3):
return num1 + num2 + num3
我們想要傳遞一個(gè)列表參數(shù)(假設(shè)這個(gè)列表有三個(gè)元素),可以這么調(diào)用函數(shù)
my_list = [1, 2, 3]
add(my_list[0], my_list[1], my_list[2])
這樣子調(diào)用要多敲幾個(gè)鍵盤,多麻煩。我們用 *
號來實(shí)現(xiàn),前面我們看到 *
可以用在函數(shù)定義的時(shí)候的用法
在函數(shù)調(diào)用的時(shí)候加一個(gè) *
號可以將一個(gè)列表或元組解構(gòu)成多個(gè)變量
my_list = [1, 2, 3]
add(*my_list)
- 雙星號(
**
)在函數(shù)中的用法
上面我們介紹了 *
操作符在函數(shù)定義的時(shí)候可以用來接收多個(gè)參數(shù),并且將參數(shù)打包成一個(gè)元組,在函數(shù)調(diào)用的時(shí)候可以對列表或者元組中的元素進(jìn)行解構(gòu)
那如果對象是字典,我們可以使用雙星號操作符(**
)
舉個(gè)例子,假設(shè)有下面的函數(shù)
def user_info(username, email, phone, date_of_birth):
user = get_user(username)
user.email = email
user.phone = phone
...
如果我們用關(guān)鍵字參數(shù)(keyword arguments,kwargs)調(diào)用它,就是下面的效果
user_info('kanye', email='blog@example.com', phone='88888', date_of_birth='2000.08.15')
那如果我需要增加傳參個(gè)數(shù),我就不得不去函數(shù)定義那里添加形參,這么做太麻煩了
我們可以這么做,我們在 kwargs
形參前面加了雙星號,用于接收任意數(shù)量的關(guān)鍵字參數(shù)
def user_info(username, **kwargs):
user = get_user(username)
user.email = kwargs['email']
user.phone = kwargs['phone']
...
在形參中,**kwargs
的關(guān)鍵是雙星號,你完全可以使用其他名字例如 **usr
(還是建議用 kwargs)
在函數(shù)定義中,**
操作符會(huì)把傳進(jìn)來的參數(shù)打包成字典
當(dāng)然,我們可以像使用 Python 中的其他字典一樣使用 kwargs
字典,這樣會(huì)使代碼更簡潔易懂
def user_info(username, **kwargs):
user = get_user(username)
for k,v in kwargs.items():
....
除此之外,**
操作符也可以用在函數(shù)調(diào)用的時(shí)候,用于將一個(gè)字典解構(gòu)成多個(gè)變量
dict = {
'email':"blog@example.com"
'phone':"88888"
...
}
user_info(username,**dict)
- 單星號接收非關(guān)鍵字參數(shù)
在函數(shù)定義的時(shí)候,單星號可以獨(dú)立使用,不需要變量(參數(shù))名稱,在Python中是一個(gè)完全有效的函數(shù)定義
def my_function(*, keyword_arg):
...
在上面的函數(shù)定義中,*
號參數(shù)會(huì)接收所有非關(guān)鍵字參數(shù)(位置參數(shù)),然后剩下的關(guān)鍵字參數(shù)傳給 keyword_arg
如果你在調(diào)用的時(shí)候只傳了位置參數(shù),就會(huì)報(bào)錯(cuò)
my_function(1)
"""
TypeError: my_function() takes 0 positional arguments but 1 was given
"""
在上面代碼中,函數(shù)my_function
定義了一個(gè)命名關(guān)鍵字參數(shù)keyword_arg
,但在函數(shù)調(diào)用時(shí)使用位置參數(shù) 1 來傳遞參數(shù),而不是使用關(guān)鍵字參數(shù)的方式,所以報(bào)錯(cuò)
函數(shù)定義中,使用*
來標(biāo)識位置參數(shù)結(jié)束,之后的參數(shù)都被視為命名關(guān)鍵字參數(shù),這意味著在函數(shù)調(diào)用時(shí)必須傳遞關(guān)鍵字參數(shù)
那如果我們想強(qiáng)制只使用位置參數(shù),而不是上面案例中的只使用關(guān)鍵字參數(shù)該怎么辦
我們使用 /
號來實(shí)現(xiàn)
def only_positional_arguments(arg1, arg2, /):
...
如果你傳遞關(guān)鍵字參數(shù),會(huì)發(fā)生報(bào)錯(cuò),是不是很有意思
only_positional_arguments(arg1=1, arg2=2)
"""
TypeError: only_positional_arguments() got some positional-only arguments passed as keyword arguments: 'arg1, arg2'
"""
- 星號拓展用法
1、單星號構(gòu)建或解構(gòu)列表
比如說有下面兩個(gè)列表
my_list_1 = [1, 2, 3]
my_list_2 = [10, 20, 30]
我們打算合并這兩個(gè)列表,可以用 +
操作符
merged_list = my_list_1 + my_list_2
但是單星號更靈活,比如說我們在合并的時(shí)候在中間加一個(gè)元素
num=42
merged_list = [*my_list_1, some_value, *my_list_2]
我們知道,解構(gòu)列表元素的時(shí)候可以這么干
my_list = [1, 2, 3]
a, b, c = my_list
# a -> 1
# b -> 2
# c -> 3
上面每個(gè)變量都存儲(chǔ)了列表中的元素,那假設(shè)我打算將列表中的第一個(gè)和最后一個(gè)元素放在指定變量中,中間的元素還是作為列表,我該怎么辦
可以使用單星號
my_list = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
a, *b, c = my_list
# a -> 1
# b -> [2, 3, 4, 5, 6, 7, 8, 9]
# c -> 10
需要注意的是,單星號接收的對象是列表
my_list = [1, 2, 3]
a, *b, c = my_list
# a -> 1
# b -> [2]
# c -> 3
2、雙星號構(gòu)建字典
同理,雙星號可以用于構(gòu)建字典
dict1 = {
'age': '22'
}
dict2 = {
'email': 'blog@example.com'
}
user_dict = {'username': 'kanye', **dict1, **dict2}