按引用传递
向函数传递参数时,采用按引用传递方式,指向时不复制实参的值到引用中,而是给他们一个新的引用。
默认参数
设置默认参数时,有几点要注意:
一是必选参数在前,默认参数在后,否则Python的解释器会报错(思考一下为什么默认参数不能放在必选参数前面);
二是如何设置默认参数:当函数有多个参数时,把变化大的参数放前面,变化小的参数放后面。变化小的参数就可以作为默认参数。
使用默认参数有什么好处?最大的好处是能降低调用函数的难度。
默认参数降低了函数调用的难度,而一旦需要更复杂的调用时,又可以传递更多的参数来实现。无论是简单调用还是复杂调用,函数只需要定义一个。
默认参数很有用,但使用不当,也会掉坑里。默认参数有个最大的坑,演示如下:
先定义一个函数,传入一个list,添加一个end再返回
def add_end(L=[]): L.append('END') return L print(add_end()) print(add_end()) print(add_end())
可以发现每次调用时记住了上一次调用时L的值
原因解释如下:
python中的默认参数只第一次定义是初始化,如果是可变变量,会导致多次调用函数时可变变量一直在变化。
要修改上面的例子,我们可以用None
这个不变对象来实现:
def add_end(L=None):
if L is None:
L = []
L.append('END')
return L
现在,无论调用多少次,都不会有问题:
>>> add_end()
['END']
>>> add_end()
['END']
不变对象一旦创建,对象内部的数据就不能修改,这样就减少了由于修改数据导致的错误。此外,由于对象不变,多任务环境下同时读取对象不需要加锁,同时读一点问题都没有。我们在编写程序时,如果可以设计一个不变对象,那就尽量设计成不变对象。
默认参数的重点是:
带默认值的参数不能位于没有默认值参数的前面。
只有在第一次调用函数时给默认参数赋值,所以要给参数附一个不变对象的值。
可变参数
定义可变参数和定义一个list或tuple参数相比,仅仅在参数前面加了一个 * 号。
要定义出这个函数,我们必须确定输入的参数。由于参数个数不确定,我们首先想到可以把数字作为一个list或tuple传进来,这样,函数可以定义如下:
def calc(numbers): sum = 0 for n in numbers: sum = sum + n * n return sum
但是调用的时候,需要先组装出一个list或tuple。
所以,我们把函数的参数改为可变参数:
def calc(*numbers): sum = 0 for n in numbers: sum = sum + n * n return sum
定义可变参数和定义一个list或tuple参数相比,仅仅在参数前面加了一个 * 号。在函数内部,参数number接收到的是一个tuple。所以仍可调用。
*num表示把这个list的所有元素作为可变参数传进去。
关键字参数
在参数前面加** 号为关键字参数。
关键字参数可以扩展函数的功能。比如,在person函数里,我们保证能接收到name和age这两个参数,但是,如果调用者愿意提供更多的参数,我们也能收到。试想你正在做一个用户注册的功能,除了用户名和年龄是必填项外,其他都是可选项,利用关键字参数来定义这个函数就能满足注册的需求。
def person(name, age, **kw): print('name:', name, 'age:', age, 'other:', kw) extra = {'city': 'Beijing', 'job': 'Engineer'} person('Jack', 24, **extra)
命名关键字参数
格式是
def person(name, age, *, city, job): print(name, age, city, job) person('Jack', 24, city='Beijing', job='Engineer')
在一般变量中加 *, 说明后面的变量为命名关键字参数
调用时必须加入关键字 city= 和 job=
否则会视为positional arguments
总结
默认参数一定要用不可变对象。
*参数 是可变参数,args接收的是一个tuple
**参数 是关键字参数,kw接收的是一个dict。
命名的关键字参数是为了限制调用者可以传入的参数名,同时可以提供默认值。