归纳起来,Python中函数的定义形式和调用形式主要有如下几种形式:
#函数的定义形式
def func(name) #匹配positional参数或者keyword参数
def func(name=value) #如果不传,有默认参数
def func(*name) #将额外的positional参数转换成一个tuple,然后赋给name
def func(**name) #将额外的keyword参数转换成一个dictionary,然后赋给name
def func(*other, name) #Python 3.X中定义keyword-only参数的形式,Python 2.X中不支持,此时,name一定要以keyword的形式传递
def func(*, name) #Python 3.X中定义keyword-only参数的形式, Python 2.X中不支持,此时, name一定要以keyword的形式传递
#函数的调用
func(name) #以positional参数调用
func(name=value) #以keyword参数调用
func(*iterable) #将iterable对象转换成一连串单个positional参数进行调用
func(**dictionary) #将dictionary转换成keywrod参数进行调用
这里需要注意3点:
1 在Python 2.X中(在Python 3.X中,这种形式已经不允许了),还有一种定义函数的方式,就是将函数的参数定义成一个tuple,那么,当调用函数的时候,传递一个结构一样的tuple参数,这个tuple参数就会被unpack:
#定义
deffunc((a, (b, c))#调用
func((1, (2, 3)) #a=1, b=2, c=3
func((1, [2, 3])) #a=1, b=2, c=3,这样调用也是可以的
func((1, 'mn')) #a=1, b='m', c='n', 这样也可以,也就是说调用的时候只要是序列都可以
#但是定义的时候只能是tuple,不能出现其他序列,比如数组,下面的情形是不对的
def func((a, [b, c]))
2 对于调用形式func(*name)和func(**name),也可以使用apply函数达到同样的效果,但是apply函数只能在Python 2.X中使用,在Python 3.X中被移除了,使用形式为:
def func(*a, **b)
a=(1, 2)
b={'c':3, 'd':4}
func(*a, **b) #Python 2.X和Python 3.X均支持
apply(func, a, b)#仅Python 2.X支持,效果一样
3 对于默认参数,尤其要注意可变对象,每一次以默认形式象调用函数,这个默认的参数会被重用,而不会重置:
def func(a=[]):
a+= [1]print(a)>>>func()
[1]>>>func() #此时a的值是[1]
[1, 1]>>>func() #此时a的值是[1, 1]
[1, 1, 1]
函数定义时参数的顺序以及函数调用时参数的顺序
Python中,定义函数和调用函数时,参数的顺序有一定的规则,如果不符合规则,Python会报错:
1 在定义函数时,参数的顺序为:positional参数(name),默认参数(name=value),*name形式(或者只有一个*,但是只在Python 3.X中支持),name或者name=value形式(都只在Python 3.X中支持,叫keyword-only参数,后面name=value是赋默认值),**name形式;
2 在调用函数时,参数的顺序为:positional参数(name),keyword参数(name=value)或者*name形式,**name形式
总之,**name无论在定义函数时,还是在调用函数时,都只应出现在最后。
函数参数的匹配步骤
Python内部,对于函数参数的匹配大致分为5步:
1 首先匹配positional参数(name)
2 匹配keywrod参数(name=value)
3 将额外的非keyword参数匹配给*name形式
4 将额外的keyword参数匹配给**name形式
5 如果有默认值,给未传递的参数赋默认值
经过上述步骤之后,Python还需要确保,每一个参数,只匹配了一个值,否则,就会报错。