打包函数参数

在函数中,你可以定义许多必需参数:

def fun1(arg1, arg2, arg3): 
    return (arg1,arg2,arg3)

这将使函数只有在给出三个参数时才可调用:

fun1(1, 2, 3)

你可以使用默认值将参数定义为可选参数:

def fun2(arg1='a', arg2='b', arg3='c'):
    return (arg1,arg2,arg3)

所以你可以用很多不同的方式调用这个函数,比如:

fun2(1)              → (1,b,c)
fun2(1, 2)           → (1,2,c)
fun2(arg2=2, arg3=3) → (a,2,3)
...

但是你也可以使用解构语法打包参数,这样你就可以使用 listdict 来分配变量。

打包参数列表

考虑一下你有一个值列表

l = [1,2,3]

你可以使用*语法将值列表作为参数调用该函数:

fun1(*l)
# Returns: (1,2,3)
fun1(*['w', 't', 'f'])
# Returns: ('w','t','f')

但是,如果你没有提供长度与参数数量匹配的列表:

fun1(*['oops'])
# Raises: TypeError: fun1() missing 2 required positional arguments: 'arg2' and 'arg3'

打包关键字参数

现在,你还可以使用字典打包参数。你可以使用**运算符告诉 Python 将 dict 解压缩为参数值:

d = {
  'arg1': 1,
  'arg2': 2,
  'arg3': 3
}
fun1(**d)
# Returns: (1, 2, 3)

当函数只有位置参数(没有默认值的那些)时,你需要包含所有预期参数的字典,并且没有额外的参数,否则你会得到一个错误:

fun1(**{'arg1':1, 'arg2':2})
# Raises: TypeError: fun1() missing 1 required positional argument: 'arg3'
fun1(**{'arg1':1, 'arg2':2, 'arg3':3, 'arg4':4})
# Raises: TypeError: fun1() got an unexpected keyword argument 'arg4'

对于具有可选参数的函数,可以使用相同的方式将参数打包为字典:

fun2(**d)
# Returns: (1, 2, 3)

但是你可以省略值,因为它们将被默认值替换:

fun2(**{'arg2': 2})
# Returns: ('a', 2, 'c')

和以前一样,你不能提供不是现有参数的额外值:

fun2(**{'arg1':1, 'arg2':2, 'arg3':3, 'arg4':4})
# Raises: TypeError: fun2() got an unexpected keyword argument 'arg4'

在实际使用中,函数可以同时具有位置和可选参数,并且它的工作原理相同:

def fun3(arg1, arg2='b', arg3='c')
    return (arg1, arg2, arg3)

你可以用一个 iterable 调用函数:

fun3(*[1])
# Returns: (1, 'b', 'c')
fun3(*[1,2,3])
# Returns: (1, 2, 3)

或者只是一本字典:

fun3(**{'arg1':1})
# Returns: (1, 'b', 'c')
fun3(**{'arg1':1, 'arg2':2, 'arg3':3})
# Returns: (1, 2, 3)

或者你可以在同一个调用中使用两个:

fun3(*[1,2], **{'arg3':3})
# Returns: (1,2,3)

请注意,你不能为同一参数提供多个值:

fun3(*[1,2], **{'arg2':42, 'arg3':3})
# Raises: TypeError: fun3() got multiple values for argument 'arg2'