args接收任意多个位置参数并打包为元组,kwargs接收任意多个关键字参数并打包为字典;二者需按“普通参数→args→仅关键字参数→kwargs”顺序定义,且常用于解包调用、装饰器和继承中。

Python 中的 *args 和 **kwargs 是处理函数参数灵活性的核心机制,它们让函数能接收不确定数量的位置参数或关键字参数,特别适合写通用工具函数、装饰器、代理调用等场景。
什么是 *args:接收任意多个位置参数
*args 用于捕获调用时传入的**多余位置参数**(即没有被前面固定形参匹配的参数),它在函数内部表现为一个 tuple。星号 * 是关键语法,表示“打包”这些参数为元组。
例如:
def greet(name, *hobbies):print(f"Hi {name}! Your hobbies: {hobbies}")
greet("Alice", "reading", "swimming", "coding")
# 输出:Hi Alice! Your hobbies: ('reading', 'swimming', 'coding')
注意:*args 必须写在所有普通参数之后、**kwargs 之前;它不强制要求叫 args,但这是广泛接受的命名惯例。
立即学习“Python免费学习笔记(深入)”;
什么是 **kwargs:接收任意多个关键字参数
**kwargs 用于捕获调用时传入的**多余关键字参数**(即未被已有形参名匹配的 key=value 参数),它在函数内部是一个 dict。双星号 ** 表示“打包成字典”。
这本书给出了一份关于python这门优美语言的精要的参考。作者通过一个完整而清晰的入门指引将你带入python的乐园,随后在语法、类型和对象、运算符与表达式、控制流函数与函数编程、类及面向对象编程、模块和包、输入输出、执行环境等多方面给出了详尽的讲解。如果你想加入 python的世界,David M beazley的这本书可不要错过哦。 (封面是最新英文版的,中文版貌似只译到第二版)
例如:
def profile(name, **info):print(f"Name: {name}")
for k, v in info.items():
print(f"{k}: {v}")
profile("Bob", age=30, city="Shanghai", job="Engineer")
# 输出:
# Name: Bob
# age: 30
# city: Shanghai
# job: Engineer
**kwargs 必须放在参数列表最后;键名必须是合法标识符,且不能与已有参数名重复,否则会报 TypeError。
组合使用 *args 和 **kwargs 的典型顺序
当一个函数需要同时支持固定参数、可变位置参数和可变关键字参数时,参数定义顺序必须严格遵守:
- 普通位置参数(如
a,b) *args- 仅关键字参数(可选,用
*隔开,如*, c=True) **kwargs
例如:
def example(a, b, *args, flag=True, **kwargs):print(f"a={a}, b={b}, args={args}, flag={flag}, kwargs={kwargs}")
调用 example(1, 2, 3, 4, flag=False, x=10, y=20) 会正确分配:a=1, b=2, args=(3, 4), flag=False, kwargs={'x': 10, 'y': 20}
实际应用中的常见技巧
除了定义函数,*args 和 **kwargs 还常用于“解包”传递参数:
- 用
*list把列表/元组元素作为位置参数传入:func(*[1,2,3])等价于func(1,2,3) - 用
**dict把字典键值对作为关键字参数传入:func(**{'x':1, 'y':2})等价于func(x=1, y=2) - 在装饰器中保留原函数签名:
def wrapper(*args, **kwargs): return func(*args, **kwargs) - 子类构造函数调用父类:
super().__init__(*args, **kwargs),避免硬编码参数名









