不使用可变参数的传统方式
我们以数学题为例子,给定一组数字 a,b … z ,请计算 sum = a * a + b * b + .....+ z * z
要定义出这个函数,我们必须确定输入的参数。由于参数个数不确定,我们首先想到可以把a, b, …, z作为一个 list 或 tuple 传进来,这样,函数可以定义如下:
def cout(numbers):sum = 0for n in numbers:sum = sum + n * nreturn sumprint(cout([1,2,3]))
14
使用可变参数
如果利用可变参数,调用函数的方式可以简化成这样:
def cout(*numbers):sum = 0for n in numbers:sum = sum + n * nreturn sumprint(cout(1, 2, 3))
14
相比于传统方式,可变参数传递方式更加简便,定义可变参数和定义一个list或tuple参数相比,仅仅在参数前面加了一个 * 号。在函数内部,参数 numbers 接收到的是一个list,因此,函数代码完全不变。但是,调用该函数时,可以传入任意个参数,包括0个参数。
如果已经有一个 list 或者 tuple,要调用一个可变参数怎么办?可以这样做:
def cout(*numbers):sum = 0for n in numbers:sum = sum + n * nreturn sum# list# my_list = [1, 2, 3]# print(cout(*my_list))# tuplemy_tuple = (1, 2, 3)print(cout(*my_tuple))
14
这种写法当然是可行的,问题是太繁琐,所以 Python 允许你在 list 或 tuple 前面加一个 * 号,把 list 或 tuple 的元素变成可变参数传进去:
nums = [1, 2, 3]print(count(*nums))
这种传递参数的方式被称为序列解包。
不可变参数传递
请看示例:
def person(name, age, **kw):print('name:', name, 'age:', age, 'other:', kw)
函数person
除了必选参数name
和age
外,还接受关键字参数**kw
。在调用该函数时,可以只传入必选参数:
方式一:
def person(name, age, **kw):print('name:', name, 'age:', age, 'other:', kw)person('Michael', 30)
name: Michael age: 30 other: {}
方式二:
也可以传入任意个数的关键字参数:
def person(name, age, **kw):print('name:', name, 'age:', age, 'other:', kw)person('Bob', 35, city='Beijing')
name: Bob age: 35 other: {'city': 'Beijing'}
方式三:
def person(name, age, **kw):print('name:', name, 'age:', age, 'other:', kw)person('Adam', 45, gender='M', job='Engineer')
name: Adam age: 45 other: {'gender': 'M', 'job': 'Engineer'}
关键字参数有什么用?它可以扩展函数的功能。比如,在person
函数里,我们保证能接收到name
和age
这两个参数,但是,如果调用者愿意提供更多的参数,我们也能收到。试想你正在做一个用户注册的功能,除了用户名和年龄是必填项外,其他都是可选项,利用关键字参数来定义这个函数就能满足注册的需求。
和可变参数类似,也可以先组装出一个dict
,然后,把该dict
转换为关键字参数传进去:
def person(name, age, **kw):print('name:', name, 'age:', age, 'other:', kw)extra = {'city': 'Beijing', 'job': 'Engineer'}person('Jack', 24, city=extra['city'], job=extra['job'])
name: Jack age: 24 other: {'city': 'Beijing', 'job': 'Engineer'}
当然,上面复杂的调用可以用简化的写法:
def person(name, age, **kw):print('name:', name, 'age:', age, 'other:', kw)extra = {'city': 'Beijing', 'job': 'Engineer'}person('Jack', 24, **extra)
name: Jack age: 24 other: {'city': 'Beijing', 'job': 'Engineer'}
**extra
表示把extra
这个dict
的所有键用关键字参数传入到函数的**kw
参数,**kw
将获得一个dict
,注意**kw
获得的dict
是extra
的一份拷贝,对**kw
的改动不会影响到函数外的extra。