Python函数参数拆解中,将可迭代对象展开为位置参数,*将字典展开为关键字参数;二者在调用时须遵循位置顺序,在定义时则用于收集多余参数,配合使用可实现灵活接口。
Python函数参数拆解的关键在于理解*和**的作用:前者把序列(如列表、元组)“摊开”成位置参数,后者把字典“摊开”成关键字参数。用错位置或类型不匹配,会直接报错。
* 后跟可迭代对象
*用于把一个可迭代对象(如列表、元组、range等)的每个元素作为单
独的位置参数传入函数。它必须出现在函数调用时参数列表的相应位置,且不能放在普通位置参数之后(除非后面还有**kwargs)。
func(*[1, 2, 3]) 等价于 func(1, 2, 3)
func(1, *[2, 3], 4) 会报错,因为*展开后不能在固定位置参数中间插入func(1, *[2, 3], **{'x': 10}) 是允许的——*在**之前,且两者都位于所有显式参数之后** 后跟字典**把字典的键值对转换为关键字参数传入。字典的键必须是字符串,且必须是合法的参数名;否则触发TypeError。
func(**{'a': 1, 'b': 2}) 等价于 func(a=1, b=2)
'c'的形参,func(**{'c': 3})会报TypeError: func() got an unexpected keyword argument 'c'
*共存:func(*args, **kwargs) 是接收任意参数的经典签名*args和**kwargs
在函数定义中,*args收集多余的位置参数为元组,**kwargs收集多余的关键字参数为字典。它们是“接收端”,而调用时的*/**是“发送端”,二者配合实现灵活接口。
def f(x, *args, y=0, **kwargs): —— x是必选位置参数,args收剩余位置参数,y是带默认值的关键字参数,kwargs收其他关键字参数f(1, 2, 3, y=5, z=6)时:x=1,args=(2,3),y=5,kwargs={'z':6}*args必须在普通参数之后、**kwargs之前;带默认值的参数若在*args后,需写成仅关键字参数形式(如*, y=0)参数拆解容易出错,推荐几种快速验证方式:
import inspect; print(inspect.signature(func)) 查看期望的参数结构print(*args)或print(**kwargs)会报错,但可以先print(args)和print(kwargs)确认内容if not isinstance(my_list, (list, tuple)): raise TypeError 避免传入不可迭代对象*[...]和**{...}会让调用难以维护,必要时封装为变量再拆解不复杂但容易忽略。掌握*和**的双向作用(定义 vs 调用),就能稳住参数传递逻辑。