和*在函数定义中是参数收集器,分别将多余位置参数和关键字参数聚合成tuple和dict;在函数调用、赋值、字典合并中才执行解包。

函数定义里 * 和 ** 是参数收集器,不是解包符号
很多人一看到 * 就默认是“解包”,其实它在函数定义中干的是相反的事:把多余的位置参数收集成一个 tuple,** 把多余的关键词参数收集成一个 dict。这是语法层面的“聚集”,不是运行时的“展开”。
-
def f(a, *args, b=10, **kwargs)中,*args捕获所有未命名的额外位置参数(如f(1, 2, 3)→args == (2, 3)) -
**kwargs只接收关键字形式的额外参数(f(1, x=100, y=200)→kwargs == {'x': 100, 'y': 200}),且必须写在参数列表末尾 - 注意顺序:
普通参数 → *args → 默认参数 → **kwargs,错序会报SyntaxError: invalid syntax - 如果只想要强制关键字参数(不允许位置传入),可写作
def g(a, *, b, c)—— 这里的*是纯分隔符,不收集参数
调用函数时 * 和 ** 才真正执行解包
这时它们的作用是把容器“摊开”成独立参数,相当于把 list、tuple 或 dict 的内容“倒进”参数槽里。
-
func(*my_list)等价于手动列出每个元素:func(my_list[0], my_list[1], ...) -
func(**my_dict)要求my_dict的键名必须是合法参数名,否则抛TypeError: func() got an unexpected keyword argument 'xxx' - 混用没问题:
func(a, *rest, **opts),但*必须在**前面,否则语法错误 - 常见误用:对生成器多次解包(
*gen)会耗尽它;若需复用,先转成list或用itertools.tee
在赋值语句中 * 实现“柔性解包”,但仅限一次
Python 3 支持在赋值左侧用 * 捕获中间或剩余项,但整个左边只能有一个 * 表达式。
-
a, *middle, z = [1, 2, 3, 4, 5]→a==1,middle==[2,3,4],z==5 -
*head, tail = 'abc'→head==['a','b'],tail=='c' - 不能写
*x, *y = [1,2,3],会报SyntaxError: two starred expressions in assignment - 空序列也能解包:
*empty, = []合法,empty得到空列表;但a, *b, c = [1]会报ValueError: not enough values to unpack
循环和推导式里慎用 * 解包,易触发 TypeError
解包不是万能钥匙,在迭代上下文中容易因类型不匹配失败。
立即学习“Python免费学习笔记(深入)”;
-
for x, *y in [[1,2], [3,4,5]]:可行,每次*y得到剩余元素组成的列表 - 但
[*x for x in ['ab', 'c']]报错:字符串不可被*直接解包(TypeError: iteration over a 0-d tensor类似提示,实际是TypeError: 'str' object is not iterable在某些旧版本) - 想展平嵌套结构,优先用
itertools.chain.from_iterable或显式循环,而不是盲目套* - 推导式中
**不可用——它只存在于函数调用和字典字面量({**d1, **d2})中,不能出现在for或if子句里
最常被忽略的一点:星号行为完全取决于它出现的语法位置——定义函数、调用函数、赋值、字典合并,四者语义不同,不能靠“记忆符号形状”来推断;出错时先看它在哪儿,再查对应规则。











