小朋友你是否有很多问号,python中有时候参数中带了和*是怎么回事?
如果是不懂或者挑刺,欢迎来阅读本篇@(滑稽)
先说最简单的序列传参
def calc(a, b, c):
return (a + b) * c
print(calc(1,5,10))
l = [1,5,10]
print(calc(*l))
*将l列表/元组等序列依次输入
而**将字典依次传入
def health_check(name, age, height, weight):
print(name)
print("您的健康状态良好")
注意键一定要是字符串,key要与参数一样
param = {"name":"张三","age":32,"height":178, "weight":85.5}
health_check(name = "张三",age = 32,height = 178, weight = 85.5)
health_check(**param)
从效果上可以看成,字典,其实是转换成形参=值传入**
但是我们经过会看到这样的参数*args,kwargs,其实作用就是上面最简单的作用,而kwargs也要格外注意!
*args
,**kwargs
def test(name, age, *args, **kwargs):
print("args:",args)
print("kwargs:",kwargs)
print("name:", name)
print("age", age)
for i in args:
print("args:", i)
for k, v in kwargs.items():
print("kwargs:", k, v)
if __name__ == '__main__':
test('yang', 18, 'is', 'a', 'good', 'man', sex="male")
result:
args: ('is', 'a', 'good', 'man')
kwargs: {'sex': 'male'}
name: yang
age 18
args: is
args: a
args: good
args: man
kwargs: sex male
我们看到,调用函数test的参数为'yang', 18, 'is', 'a', 'good', 'man', sex="male",python依次将yang和18赋给了name和age,将'is', 'a', 'good', 'man'赋给了args元组,将sex="male"转换成了一个字典给了kwargs.
你可以认为它们收了多余的参数,并且重新转元组和字典。
那好,**kwargs就会把字典转成形参=...的样子(这在前面提到了),直接输出kwargs得到的是字典
于是这就诞生了一个问题,
为什么print(*args)
可以输出
而print(**kwargs)
却报错 'name' is an invalid keyword argument for this function
正是因为**kwargs相当于形参=...,于是相当于传入print(形参=...),可是print没有这个形参啊,因此报错
所以可以说**kwargs
就是形参=值,<===> 形参=值转换字典存于kwargs这个变量中
*args
就是值 <===> 值转成元组存agrs这个变量中
继续,我们往往还在 装饰器 中看到*args与**kwargs
def log(name=None):
def decorator(func):
"""如果函数func是带参数,那么就得加入魔法参数,把函数的参数传入"""
"""*args传的是元组,**kwargs传的是字典"""
def wrapper(*args, **kwargs):
print('{0}.start'.format(name))
# print(args)
# print(kwargs)
rest = func(*args, **kwargs)
print('{0}.end'.format(name))
""" func 有返回值,也要返回"""
return rest
return wrapper
return decorator
@log("from add")
def add(a, b):
return a + b
通过*args和**kwargs
传值,而args和kwargs就跟上面存的一直,自行组合成元组或者字典,并且通过* 和 **
转换....(不再重复)
def test(name,age,*args,**kwargs):
print("args:", args)
print("kwargs:",kwargs)
print("name:",name)
print('age:',age)
if __name__ == '__main__':
d = {"name1":"杨彦星",'age1':18}
l = ['a','b','c']
t = ('d','e','f')
test(*l,*t,**d) # 看参数位置
args: ('c', 'd', 'e', 'f')
kwargs: {'age1': 18, 'name1': '杨彦星'}
name: a
age: b
这里有几个问题需要注意下,由于**kwargs
只能定义在函数参数的最后,它后面不能再有形参了,但是*参数可以不限制参数的位置,所以上面的代码在定义d的时候不能再有name和age的定义了,因为在传参的时候,已经将’a’赋给了name,’b’赋给了age,所以如果之后再有name的定义则会报参数重复定义的错误。但是却可以如下的定义
def test(*args,name,age,**kwargs):
print("args:", args)
print("kwargs:",kwargs)
print("name:",name)
print('age:',age)
if __name__ == '__main__':
d = {"name":"杨彦星",'age':18,'sex':'male'}
l = ['a','b','c']
t = ('d','e','f')
test(*l,*t,**d)
因为把*args放到了第一个参数,所以在调用函数时为test('a','b','c','d','e','f',name="杨彦星",age=18,sex='male'),此时如果d的定义的时候没有name和age也会报错。
好吧,也就是说写到这里,*args
也有重要点,它表示值,如果没有像字典,就会把值代入默认的形参中**