Python函数的可变参数
2015-03-10 09:37
537 查看
刚学用Python的时候,特别是看一些库的源码时,经常会看到func(*args, **kwargs)这样的函数定义,这个*和**让人有点费解。其实只要把函数参数定义搞清楚了,就不难理解了。
先说说函数定义,我们都知道,下面的代码定义了一个函数funcA
def funcA():
pass
显然,函数funcA没有参数(同时啥也不干:D)。
下面这个函数funcB就有两个参数了,
def funcB(a, b):
print a
print b
调用的时候,我们需要使用函数名,加上圆括号扩起来的参数列表,比如 funcB(100, 99),执行结果是:
100
99
很明显,参数的顺序和个数要和函数定义中一致,如果执行funcB(100),Python会报错的:
TypeError: funcB() takes exactly 2 arguments (1 given)
我们可以在函数定义中使用参数默认值,比如
def funcC(a, b=0):
print a
print b
在函数funcC的定义中,参数b有默认值,是一个可选参数,如果我们调用funcC(100),b会自动赋值为0。
OK,目前为止,我们要定义一个函数的时候,必须要预先定义这个函数需要多少个参数(或者说可以接受多少个参数)。一般情况下这是没问题的,但是也有在定义函数的时候,不能知道参数个数的情况(想一想C语言里的printf函数),在Python里,带*的参数就是用来接受可变数量参数的。看一个例子
def funcD(a, b, *c):
print a
print b
print "length of c is: %d " % len(c)
print c
调用funcD(1, 2, 3, 4, 5, 6)结果是
1
2
length of c is: 4
(3, 4, 5, 6)
我们看到,前面两个参数被a、b接受了,剩下的4个参数,全部被c接受了,c在这里是一个tuple。我们在调用funcD的时候,至少要传递2个参数,2个以上的参数,都放到c里了,如果只有两个参数,那么c就是一个empty tuple。
好了,一颗星我们弄清楚了,下面轮到两颗星。
上面的例子里,调用函数的时候,传递的参数都是根据位置来跟函数定义里的参数表匹配的,比如funcB(100, 99)和funcB(99, 100)的执行结果是不一样的。在Python里,还支持一种用关键字参数(keyword argument)调用函数的办法,也就是在调用函数的时候,明确指定参数值付给那个形参。比如还是上面的funcB(a, b),我们通过这两种方式调用
funcB(a=100, b=99)
和
funcB(b=99, a=100)
结果跟funcB(100, 99)都是一样的,因为我们在使用关键字参数调用的时候,指定了把100赋值给a,99赋值给b。也就是说,关键字参数可以让我们在调用函数的时候打乱参数传递的顺序!
另外,在函数调用中,可以混合使用基于位置匹配的参数和关键字参数,前题是先给出固定位置的参数,比如
def funcE(a, b, c):
print a
print b
print c
调用funcE(100, 99, 98)和调用funcE(100, c=98, b=99)的结果是一样的。
好了,经过以上铺垫,两颗星总算可以出场了:
如果一个函数定义中的最后一个形参有 ** (双星号)前缀,所有正常形参之外的其他的关键字参数都将被放置在一个字典中传递给函数,比如:
def funcF(a, **b):
print a
for x in b:
print x + ": " + str(b[x])
调用funcF(100, c='你好', b=200),执行结果
100
c: 你好
b: 200
大家可以看到,b是一个dict对象实例,它接受了关键字参数b和c。
同C语言一样,Python中也有可变参数函数,即一个函数可以接收多个参数,而这些参数的个数在函数调用之前事先是不知道的。
在C里面最典型的例子就是printf函数,同样在python里面使用可变参数也可以很方便的实现printf()
代码如下:
[python] view
plaincopy
def printf(fmt, *arg):
print fmt % arg
printf定义中的*arg就是可变参数,它的类型为tuple,代表了除fmt参数之外的所有传入参数。
但如果我们传入的可变参数是字典,那么*arg的定义就不够用了,这时我们应该使用**arg来定义函数。
这时我们调用函数的方式变为:
定义:def fun(**arg)
fun(key1 = val1, key2=val2, ...)
如: fun(One=1, Two=2)
代码如下:
[python] view
plaincopy
def testFun1(**arg):
for k in arg.keys():
printf("arg[%s] = %s.", k, arg[k])
def testFun2(*arg1, **arg2):
print arg1
print arg2
if __name__ == '__main__':
printf("How do you do? %d, %d", 2, 3)
testFun1(One=1, Two=2)
testFun2(1, 2, 3, One=1, Two=2, Three=3)
输出结果:
How do you do? 2, 3
arg[Two] = 2.
arg[One] = 1.
(1, 2, 3)
{'Three': 3, 'Two': 2, 'One': 1}
先说说函数定义,我们都知道,下面的代码定义了一个函数funcA
def funcA():
pass
显然,函数funcA没有参数(同时啥也不干:D)。
下面这个函数funcB就有两个参数了,
def funcB(a, b):
print a
print b
调用的时候,我们需要使用函数名,加上圆括号扩起来的参数列表,比如 funcB(100, 99),执行结果是:
100
99
很明显,参数的顺序和个数要和函数定义中一致,如果执行funcB(100),Python会报错的:
TypeError: funcB() takes exactly 2 arguments (1 given)
我们可以在函数定义中使用参数默认值,比如
def funcC(a, b=0):
print a
print b
在函数funcC的定义中,参数b有默认值,是一个可选参数,如果我们调用funcC(100),b会自动赋值为0。
OK,目前为止,我们要定义一个函数的时候,必须要预先定义这个函数需要多少个参数(或者说可以接受多少个参数)。一般情况下这是没问题的,但是也有在定义函数的时候,不能知道参数个数的情况(想一想C语言里的printf函数),在Python里,带*的参数就是用来接受可变数量参数的。看一个例子
def funcD(a, b, *c):
print a
print b
print "length of c is: %d " % len(c)
print c
调用funcD(1, 2, 3, 4, 5, 6)结果是
1
2
length of c is: 4
(3, 4, 5, 6)
我们看到,前面两个参数被a、b接受了,剩下的4个参数,全部被c接受了,c在这里是一个tuple。我们在调用funcD的时候,至少要传递2个参数,2个以上的参数,都放到c里了,如果只有两个参数,那么c就是一个empty tuple。
好了,一颗星我们弄清楚了,下面轮到两颗星。
上面的例子里,调用函数的时候,传递的参数都是根据位置来跟函数定义里的参数表匹配的,比如funcB(100, 99)和funcB(99, 100)的执行结果是不一样的。在Python里,还支持一种用关键字参数(keyword argument)调用函数的办法,也就是在调用函数的时候,明确指定参数值付给那个形参。比如还是上面的funcB(a, b),我们通过这两种方式调用
funcB(a=100, b=99)
和
funcB(b=99, a=100)
结果跟funcB(100, 99)都是一样的,因为我们在使用关键字参数调用的时候,指定了把100赋值给a,99赋值给b。也就是说,关键字参数可以让我们在调用函数的时候打乱参数传递的顺序!
另外,在函数调用中,可以混合使用基于位置匹配的参数和关键字参数,前题是先给出固定位置的参数,比如
def funcE(a, b, c):
print a
print b
print c
调用funcE(100, 99, 98)和调用funcE(100, c=98, b=99)的结果是一样的。
好了,经过以上铺垫,两颗星总算可以出场了:
如果一个函数定义中的最后一个形参有 ** (双星号)前缀,所有正常形参之外的其他的关键字参数都将被放置在一个字典中传递给函数,比如:
def funcF(a, **b):
print a
for x in b:
print x + ": " + str(b[x])
调用funcF(100, c='你好', b=200),执行结果
100
c: 你好
b: 200
大家可以看到,b是一个dict对象实例,它接受了关键字参数b和c。
同C语言一样,Python中也有可变参数函数,即一个函数可以接收多个参数,而这些参数的个数在函数调用之前事先是不知道的。
在C里面最典型的例子就是printf函数,同样在python里面使用可变参数也可以很方便的实现printf()
代码如下:
[python] view
plaincopy
def printf(fmt, *arg):
print fmt % arg
printf定义中的*arg就是可变参数,它的类型为tuple,代表了除fmt参数之外的所有传入参数。
但如果我们传入的可变参数是字典,那么*arg的定义就不够用了,这时我们应该使用**arg来定义函数。
这时我们调用函数的方式变为:
定义:def fun(**arg)
fun(key1 = val1, key2=val2, ...)
如: fun(One=1, Two=2)
代码如下:
[python] view
plaincopy
def testFun1(**arg):
for k in arg.keys():
printf("arg[%s] = %s.", k, arg[k])
def testFun2(*arg1, **arg2):
print arg1
print arg2
if __name__ == '__main__':
printf("How do you do? %d, %d", 2, 3)
testFun1(One=1, Two=2)
testFun2(1, 2, 3, One=1, Two=2, Three=3)
输出结果:
How do you do? 2, 3
arg[Two] = 2.
arg[One] = 1.
(1, 2, 3)
{'Three': 3, 'Two': 2, 'One': 1}
相关文章推荐
- Python中函数的参数定义和可变参数
- [Python] python中函数的参数定义与可变参数
- Python中函数的参数定义和可变参数
- Python函数可变参数定义及其参数传递方式实例详解
- 【转】Python可变长度的函数参数
- Python中函数的参数定义和可变参数
- Python中函数的参数定义和可变参数
- Python——可变类型与不可变类型(即为什么函数默认参数要用元组而非列表)
- Python函数可变参数定义及其参数传递方式详解
- Python函数可变参数定义及其参数传递方式实例详解
- Python中函数个数可变参数的传递
- python函数参数是值传递还是引用传递(以及变量间复制后是否保持一致):取决于对象内容可变不可变
- python函数参数是值传递还是引用传递(以及变量间复制后是否保持一致):取决于对象内容可变不可变
- Noah的学习笔记之Python篇:函数“可变长参数”
- Python中函数的参数定义和可变参数
- Python中函数的参数定义和可变参数
- 转:Python中函数的参数定义和可变参数
- Python中函数的参数传递与可变长参数
- Python中函数的参数定义和可变参数
- Python中函数的参数定义和可变参数