总结:
普通形参就是一个变量名,可以使用位置实参或者关键字实参。*args 形参可以吸收任意多个位置实参,必须用位置实参进行传递实参,实参被封装进一个元组。**kwargs 形参可以吸收任意多个关键字实参,必须使用关键字实参进行传递实参,实参被封装进一个字典,每一对关键字实参的左值作为键,右值作为值。
定义函数时,在形参列表里,普通形参、* 形参和 ** 形参可以混用,要注意两点:1.各自的个数。2.相互之间的位置。
1.关于各自的个数:普通形参可以有多个,但 * 形参和 ** 形参只能有一个。因为* 形参和 ** 形参都可以吸收任意多个实参,如果有多个这样的形参,将不知道哪些实参属于哪个形参,引发混乱。
2.关于相互之间的位置:形参位置只要保证两点则形参位置顺序就是正确的:(1)保证实参列表中位置实参在关键字实参前。(2)实参列表中后一个形参对应的实参不会被前一个形参吸收为前一个形参的实参,从而引起混乱。
这是一个收集参数的过程,也是打包的过程,相对应的是解包,也叫逆向收集参数。
关于形参的个数问题举例
下面是错误的,因为实参 3 和 4 后面的其他实参不知道哪些属于形参 books,哪些属于形参 ok。有多个 **kwargs 的情况同理。
def test(num, go, *books, *ok):
print('num:', num)
print('books:', books)
test(3, 4, 'python', 'java', 'c','oo','pp')
报错:SyntaxError: invalid syntax
关于形参位置的问题举例
*args 和普通形参
1、位于 args 前面的普通形参只能使用位置实参,如果使用关键字实参,而args 形参只能使用位置实参,那么实参列表会出现位置实参位于关键字实参后面的情况,会报错。
2、位于 *args 后面的普通形参需要使用关键字实参传值,如果使用位置实参会被吸收进前的 *args 形参。
def test(*books, num):
print('books:', books)
print('num:', num)
test('python', 'java', 'c', 3)
报错:TypeError: test() missing 1 required keyword-only argument: ‘num’
**kwargs 和普通形参
3、普通形参可以在 **kwargs 前面,普通形参可以使用位置实参或者关键字实参。
def test(num, **scores):
print('num:', num)
print('scores:', scores)
test(3, py=90, ja=80)
def test(num, **scores):
print('num:', num)
print('scores:', scores)
test(num=3, py=90, ja=80)
4、普通形参不能放在 **kwargs 后面,如果普通形参使用位置实参会导致实参列表中位置实参位于关键字实参后面,如果普通形参使用关键字实参会被前面的 **kwargs 吸收。
*args 和 **kwargs
5、*args 要放在 **kwargs 前面,因为实参列表里面位置实参要在关键字实参前面。
示例和补充:
def foo(*args,**kwargs):
print ('args=',args)
print ('kwargs=',kwargs)
print ('*'*20)
if __name__=='__main__':
foo(1,2,3)
foo(a=1,b=2,c=3)
foo(1,2,3,a=1,b=2,c=3)
foo(1,'b','c',a=1,b='b',c='c')
运行结果:
args= (1, 2, 3)
kwargs= {}
********************
args= ()
kwargs= {'a': 1, 'b': 2, 'c': 3}
********************
args= (1, 2, 3)
kwargs= {'a': 1, 'b': 2, 'c': 3}
********************
args= (1, 'b', 'c')
kwargs= {'a': 1, 'b': 'b', 'c': 'c'}
********************
|