對于python函數(shù)參數(shù),對于初學者可能就是進入了迷宮,盡管我也是初學者,簡單總結(jié)一下。
說參數(shù)之前,先講一下兩個packing(包裹)和unpacking(解包裹):
a = (1,2,3)
b,c,d = a
print(b)
print(c)
print(d)
輸出:
1
2
3
我總結(jié)不了這個概念,只能幫大家到這了
一、位置參數(shù)和關鍵字參數(shù):
調(diào)用函數(shù)時根據(jù)函數(shù)定義的參數(shù)位置來傳遞參數(shù)。
def print_hello(name, sex):
sex_dict = {1: '先生', 2: '女士'}
print 'hello %s %s, welcome to python world!' %(name, sex_dict.get(sex, '先生'))
# 兩個參數(shù)的順序必須一一對應,且少一個參數(shù)都不可以
# print_hello('李四', 1)
# 當然你可以通過強制指定關鍵字來更換參數(shù)
# print_hello(sex=1,name='李四')
注意:
有位置參數(shù)時,位置參數(shù)必須在關鍵字參數(shù)的前面,但關鍵字參數(shù)之間不存在先后順序的
二、默認參數(shù):
用于定義函數(shù),為參數(shù)提供默認值,調(diào)用函數(shù)時可傳可不傳該默認參數(shù)的值(注意:所有位置參數(shù)必須出現(xiàn)在默認參數(shù)前,包括函數(shù)定義和調(diào)用)
# 正確的默認參數(shù)定義方式--> 位置參數(shù)在前,默認參數(shù)在后
def print_hello(name, sex=1):
....
# 錯誤的定義方式
def print_hello(sex=1, name):
....
# 調(diào)用時不傳sex的值,則使用默認值1
# print_hello('tanggu')
# 調(diào)用時傳入sex的值,并指定為2
# print_hello('tanggu', 2)
三、可變參數(shù):
定義函數(shù)時,有時候我們不確定調(diào)用的時候會傳遞多少個參數(shù)(不傳參也可以)。此時,可用包裹(packing)位置參數(shù),或者包裹關鍵字參數(shù),來進行參數(shù)傳遞,會顯得非常方便。
1、包裹位置傳遞
def func(*args):
....
# func()
# func(a)
# func(a, b, c)
我們傳進的所有參數(shù)都會被args變量收集,它會根據(jù)傳進參數(shù)的位置合并為一個元組(tuple),args是元組類型,這就是包位置傳遞。
2、包裹關鍵字傳遞
def func(**kargs):
....
# func(a=1)
# func(a=1, b=2, c=3)
kargs是一個字典(dict),收集所有關鍵字參數(shù)
四、解包裹參數(shù):
*args和 **kargs,也可以在函數(shù)調(diào)用的時候使用,稱之為解包(unpacking)
1、在傳遞元組時,讓元組的每一個元素對應一個位置參數(shù)
def print_hello(name, sex):
print name, sex
# args = ('tanggu', '男')
# print_hello(*args)
# tanggu 男
2、在傳遞詞典字典時,讓詞典的每個鍵值對作為一個關鍵字參數(shù)傳遞給函數(shù)
def print_hello(kargs):
print kargs
# kargs = {'name': 'tanggu', 'sex', u'男'}
# print_hello(**kargs)
# {'name': 'tanggu', 'sex', u'男'}
五、位置參數(shù)、默認參數(shù)、可變參數(shù)的混合使用
1、基本原則是:先位置參數(shù),默認參數(shù),包裹位置,包裹關鍵字(定義和調(diào)用都應遵循)
def func(name, age, sex=1, *args, **kargs):
print name, age, sex, args, kargs
# func('tanggu', 25, 2, 'music', 'sport', class=2)
# tanggu 25 1 ('music', 'sport') {'class'=2}
2、Python中 *args 和**kwargs的區(qū)別
先看個demo:
def foo(*args, **kwargs):
print 'args = ', args
print 'kwargs = ', kwargs
print '---------------------------------------'
if __name__ == '__main__':
foo(1,2,3,4)
foo(a=1,b=2,c=3)
foo(1,2,3,4, a=1,b=2,c=3)
foo('a', 1, None, a=1, b='2', c=3)
輸出結(jié)果:
args = (1, 2, 3, 4)
kwargs = {}
---------------------------------------
args = ()
kwargs = {'a': 1, 'c': 3, 'b': 2}
---------------------------------------
args = (1, 2, 3, 4)
kwargs = {'a': 1, 'c': 3, 'b': 2}
---------------------------------------
args = ('a', 1, None)
kwargs = {'a': 1, 'c': 3, 'b': '2'}
分析一下:可以看到,這兩個是[Python]中的可變參數(shù)。*args 表示任何多個無名參數(shù),它是一個tuple;**kwargs 表示關鍵字參數(shù),它是一個dict。并且同時使用*args和**kwargs時,必須*args參數(shù)列要在**kwargs前,否則會報語法錯誤!!!
還有個小應用場景:創(chuàng)建字典
def kw_dict(**kwargs):
return kwargs
print kw_dict(a=1,b=2,c=3) == {'a':1, 'b':2, 'c':3}
其實python中就帶有dict類,使用dict(a=1,b=2,c=3)即可創(chuàng)建一個字典了。
*args:
重點在*,后面的args相當于一個變量名,可以自己定義的。它的本質(zhì)就是將標準調(diào)用剩下的值集中轉(zhuǎn)變?yōu)樵M。
#*args的運用
def foo (x,*args):
print(x)
print(args)
foo(1,2,3,4,5,6,6,'a','b') #調(diào)用函數(shù)
輸出:
1----->x
(2, 3, 4, 5, 6, 6, 'a', 'b')----->*args
從形參的角度:
def foo(*args): # foo(x,y,z)參數(shù)個數(shù)可以至多個
print(args)
foo(1, 2, 3) # 沒有個數(shù)限制
(1, 2, 3) #輸出結(jié)果<br>#*args=*(1,2,3)
從實參的角度:
def bar(x,y,z):
print(x)
print(y)
print(z)
# bar(*(1,2,3,4)) #會報錯只需要3個值,但給定的實參有4個值,不能一一對應
bar(*(1,2,3)) #實參一一對應
2從實參的角度
def bar(x,y,z):
print(x)
print(y)
print(z)
# bar(*(1,2,3,4)) #會報錯只需要3個值,但給定的實參有4個值,不能一一對應
bar(*(1,2,3)) #實參一一對應
從不同角度看**kwargs:
def foo(**kwargs): #foo(x=1,y=2,z=3)參數(shù)個數(shù)可以一個至多個
print(kwargs)
foo(x=1,y=2,z=3) #調(diào)用函數(shù),沒有個數(shù)限制
{'x': 1, 'y': 2, 'z': 3} ----->輸出字典
--------------分割線-------------------------
def foo(x,y,z):
print(x)
print(y)
print(z)
# foo(**{'x' : 1,'y' : 2, 'z' : 3,"a":4 })#TypeError: foo() got an unexpected keyword argument 'a',報錯原因是函數(shù)foo只有3個值(x,y,z)
foo(**{'x' : 1,'y' : 2, 'z' : 3}) #三個值正好匹配
**kwargs與位置參數(shù)和默認參數(shù)混用:
def foo(x,a = 4,**kwargs): #混合使用參數(shù)
print(x)
print(a)
print(kwargs)
foo(1,y=2,z=3) #使用默認參數(shù)
#輸出
{'y': 2, 'z': 3}
foo(1,5,y=2,z=3) #修改默認參數(shù)
5
{'y': 2, 'z': 3}
超復雜混合參數(shù)混用記:
def foo(x,*args,a=4,**kwargs): #使用默認參數(shù)時,注意默認參數(shù)的位置要在args之后kwargs之前
print(x)
print(a)
print(args)
print(kwargs)
foo(1,5,6,7,8,y=2,z=3) #調(diào)用函數(shù),不修改默認參數(shù)
#x的值
#a的值
(5, 6, 7, 8) #*args的值
{'y': 2, 'z': 3} ##kwargs的值
def foo(x,a=4,*args,**kwargs): ##注意:當需要修改默認參數(shù)時,要調(diào)整默認參數(shù)的位置,要放在args之前即可,但不可放在開頭。
print(x)
print(a)
print(args)
print(kwargs)
foo(1,9,5,6,7,8,y=2,z=3) #調(diào)用函數(shù),修改默認參數(shù)a為9
#x的值
#被修改后a的值
(5, 6, 7, 8) #args的值
{'y': 2, 'z': 3} #kwargs的值
總結(jié):
位置參數(shù):
調(diào)用函數(shù)時所傳參數(shù)的位置必須與定義函數(shù)時參數(shù)的位置相同
關鍵字參數(shù):
使用關鍵字參數(shù)會指定參數(shù)值賦給哪個形參,調(diào)用時所傳參數(shù)的位置可以任意
*位置參數(shù):可接受任意數(shù)量的位置參數(shù)(元組);只能作為最后一個位置參數(shù)出現(xiàn),其后參數(shù)均為關鍵字參數(shù)
**關鍵字參數(shù):可接受任意數(shù)量的關鍵字參數(shù)(字典);只能作為最后一個參數(shù)出現(xiàn)