python的位置引數、預設引數、關鍵字引數、可變引數區別
一、位置引數
呼叫函式時根據函式定義的引數位置來傳遞引數。
#!/usr/bin/env python # coding=utf-8 def print_hello(name, sex): sex_dict = {1: u'先生', 2: u'女士'} print 'hello %s %s, welcome to python world!' %(name, sex_dict.get(sex, u'先生')) # 兩個引數的順序必須一一對應,且少一個引數都不可以 # print_hello('tanggu', 1)
二、關鍵字引數
用於函式呼叫,通過“鍵-值”形式加以指定。可以讓函式更加清晰、容易使用,同時也清除了引數的順序需求。
# 以下是用關鍵字引數正確呼叫函式的例項 # print_hello('tanggu', sex=1) # print_hello(1, name='tanggu') # print_hello(name='tanggu', sex=1) # print_hello(sex=1, name='tanggu') # 以下是錯誤的呼叫方式 # print_hello(name='tanggu', 1) # print_hello(sex=1, 'tanggu')
通過上面的程式碼,我們可以發現:有位置引數時,位置引數必須在關鍵字引數的前面,但關鍵字引數之間不存在先後順序的
三、預設引數
用於定義函式,為引數提供預設值,呼叫函式時可傳可不傳該預設引數的值(注意:所有位置引數必須出現在預設引數前,包括函式定義和呼叫)
# 正確的預設引數定義方式--> 位置引數在前,預設引數在後 def print_hello(name, sex=1): .... # 錯誤的定義方式 def print_hello(sex=1, name): .... # 呼叫時不傳sex的值,則使用預設值1 # print_hello('tanggu') # 呼叫時傳入sex的值,並指定為2 # print_hello('tanggu', 2)
四、可變引數
定義函式時,有時候我們不確定呼叫的時候會傳遞多少個引數(不傳參也可以)。此時,可用包裹(packing)位置引數,或者包裹關鍵字引數,來進行引數傳遞,會顯得非常方便。
1、包裹位置傳遞
def func(*args): .... # func() # func(a) # func(a, b, c)
我們傳進的所有引數都會被args變數收集,它會根據傳進引數的位置合併為一個元組(tuple),args是元組型別,這就是包裹位置傳遞。
2、包裹關鍵字傳遞
def func(**kargs): .... # func(a=1) # func(a=1, b=2, c=3)
kargs是一個字典(dict),收集所有關鍵字引數
五、解包裹引數
*和**,也可以在函式呼叫的時候使用,稱之為解包裹(unpacking)
1、在傳遞元組時,讓元組的每一個元素對應一個位置引數
def print_hello(name, sex): print name, sex # args = ('tanggu', '男') # print_hello(*args) # tanggu 男
2、在傳遞詞典字典時,讓詞典的每個鍵值對作為一個關鍵字引數傳遞給函式
def print_hello(kargs): print kargs # kargs = {'name': 'tanggu', 'sex', u'男'} # print_hello(**kargs) # {'name': 'tanggu', 'sex', u'男'}
六、位置引數、預設引數、可變引數的混合使用
基本原則是:先位置引數,預設引數,包裹位置,包裹關鍵字(定義和呼叫都應遵循)
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}
先來看個例子:
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)
輸出結果如下:
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中的可變引數。*args
表示任何多個無名引數,它是一個tuple;**kwargs
表示關鍵字引數,它是一個dict。並且同時使用*args
和**kwargs
時,必須*args
引數列要在**kwargs
前,像foo(a=1, b='2', c=3, a', 1, None, )
這樣呼叫的話,會提示語法錯誤“SyntaxError: non-keyword arg after keyword arg”
。
知道*args
和`**kwarg是什麼了吧。還有一個很漂亮的用法,就是建立字典:
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)即可建立一個字典了。
*args:
重點在*,後面的args相當於一個變數名,可以自己定義的。它的本質就是將標準呼叫剩下的值集中轉變為元組。
#*args的運用 def foo (x,*args): print(x) print(args) foo(1,2,3,4,5,6,6,'a','b') #呼叫函式 輸出:
1----->x
(2, 3, 4, 5, 6, 6, 'a', 'b')----->*args
1 2 3 4 5 6 |
|
從實參的角度 def bar(x,y,z): print(x) print(y) print(z) # bar(*(1,2,3,4)) #會報錯只需要3個值,但給定的實參有4個值,不能一一對應 bar(*(1,2,3)) #實參一一對應 1 2 3
*args與位置引數和預設引數混用:(輸出元組)
#*args與位置引數和預設引數混用:*args要放到位置引數後面,預設引數要放最後。 def foo(x,*args,y = 1): print(x) print(y) print(args) foo(1,2,3,4,5,6,7,8,9,10,y=10000) #呼叫函式 1------->x=1 10000-------->y=10000 (2, 3, 4, 5, 6, 7, 8, 9, 10)----->args的值變成元組
**kwargs(輸出字典)
重點在**,後面的kwargs相當於一個變數名,可以自己定義的。它的本質就是將關鍵字呼叫剩下的值集中轉變為字典。 def foo(x,**kwargs): #kwargs = {'y':1,'z': 2} print(x) print(kwargs) foo(1,y=1,z=2) #呼叫函式 # 1-->x的值 # {'y': 1, 'z': 2} #kwargs的值,轉為了一個字典
從不同角度看**kwargs:
def foo(**kwargs): #foo(x=1,y=2,z=3)引數個數可以一個至多個 print(kwargs) foo(x=1,y=2,z=3) #呼叫函式,沒有個數限制 {'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',報錯原因是函式foo只有3個值(x,y,z) foo(**{'x' : 1,'y' : 2, 'z' : 3}) #三個值正好匹配 1#輸出結果 2 3
**kwargs與位置引數和預設引數混用:
def foo(x,a = 4,**kwargs): #混合使用引數 print(x) print(a) print(kwargs) foo(1,y=2,z=3) #使用預設引數
1 #輸出
4
{'y': 2, 'z': 3}
foo(1,5,y=2,z=3) #修改預設引數
1
5
{'y': 2, 'z': 3}
超複雜混合引數混用記:
從形參角度:
def foo(x,*args,a=4,**kwargs): #使用預設引數時,注意預設引數的位置要在args之後kwargs之前 print(x) print(a) print(args) print(kwargs) foo(1,5,6,7,8,y=2,z=3) #呼叫函式,不修改預設引數 1 #x的值 4 #a的值 (5, 6, 7, 8) #*args的值 {'y': 2, 'z': 3} ##kwargs的值
def foo(x,a=4,*args,**kwargs): ##注意:當需要修改預設引數時,要調整預設引數的位置,要放在args之前即可,但不可放在開頭。 print(x) print(a) print(args) print(kwargs) foo(1,9,5,6,7,8,y=2,z=3) #呼叫函式,修改預設引數a為9 1 #x的值 9 #被修改後a的值 (5, 6, 7, 8) #args的值 {'y': 2, 'z': 3} #kwargs的值
從實參角度:
def foo(x,*args,a=4,**kwargs): #使用預設引數時,注意預設引數的位置要在args之後kwargs之前 print(x) print(a) print(args) print(kwargs) foo(1,*(5,6,7,8),**{"y":2,"z":3}) #呼叫函式,不改預設引數 1 #x的值 4 #預設a的值 (5, 6, 7, 8) #args的值 {'y': 2, 'z': 3} #kwargs的值 -------------分割線----------------------
def foo(x,a=4,*args,**kwargs): #注意:當需要修改預設引數時,要調整預設引數的位置,要放在args之前,但不可放在開頭 print(x) print(a) print(args) print(kwargs) foo(1,9,10,*(5,6,7,8),**{"y":2,"z":3}) #呼叫函式,修改預設引數a為9 1 #x的值 9 #修改預設引數a後的值 (10, 5, 6, 7, 8) #args的值 {'y': 2, 'z': 3} #kwargs的值
小結:
位置引數:
呼叫函式時所傳引數的位置必須與定義函式時引數的位置相同
關鍵字引數:
使用關鍵字引數會指定引數值賦給哪個形參,呼叫時所傳引數的位置可以任意
*位置引數:可接受任意數量的位置引數(元組);只能作為最後一個位置引數出現,其後引數均為關鍵字引數
**關鍵字引數:可接受任意數量的關鍵字引數(字典);只能作為最後一個引數出現
預設引數:預設引數的賦值只會在函式定義的時候繫結一次,預設值不會再被修改