Python基础之定义有默认参数的函数

1. 构建有默认参数的函数

当我们在构建一个函数或者方法时,如果想使函数中的一个或者多个参数使可选的,并且有一个默认值,那么可以在函数定义中给参数指定一个默认值,并且放到参数列表的最后就行了。比如:

def func(a, b=42):
    print(a, b)
func(1)             #a=1, b=42 
func(1,2)           #a=1, b=2

如果默认参数使一个可以修改的容器,比如一个列表,集合或者字典,可以使用None作为默认值。比如:

#使用列表list作为默认值
def func(a, b=None):
    if b is None:
        b = []
    ···

但是,如果你并不是想提供一个默认值,而仅仅知识想测试下某个默认参数是不是有值传递进来,可以这样写:

_no_value = object()
def func(a, b=_no_value):
    if b is _no_value:
        print("b没有值")
    else:
        print(a, b)

func(1)                     # b没有值
func(1, None)               # 1 None

仔细通过测试可以发现,传递一个None值和不传值两种情况是有差别的。

2. 参数陷阱

2.1 默认参数的值

默认参数的值仅仅在函数定义的时候赋值一次,示例代码如下:

x = 42
def func(a, b=x):
    print(a, b)

func(1)                    #1 42
x = 23
func(1)                    #1 42

从上面例子中可以看出,当我们改变x的值的对默认参数值并没有影响,这是因为在函数定定义的时候就已经确定了它的默认值了。

2.2 默认参数值的类型

默认参数值的类型应该是不可变对象,比如None,True,False,数字或字符串,而不能使用列表,字典等可变类型。

不要像下面这样写代码:

def func(a, b=[]):          #不能这么写
    ···

如果这样写了,当默认值在某些地方被修改之后,程序就会出现问题。

这些被修改的程序会影响到下次调用这个参数时的默认值。比如:

def func(a, b=[]):
    return b
x = func(1)
print(x)                #[]
x.append("a")
x.append("b")
print(x)                #['a', 'b']
y = func(1)
print(y)                #['a', 'b']

可以看到,b的默认值从一个空list,变成了[“a”, “b”],这种不会是你想要的默认参数。

为了避免这种情况的发生,最好是将默认值设置为None,然后在函数里面检查它。

但是在将默认值设置为None之后,测试None值时使用is操作符是很重要的,is是这种方法的关键点。
有人可能会这样写:

def func(a, b=None):
    if not b:
        b = []
···

这么写有有一个问题,就是虽然None确实会被当成False,但是还有其他的对象(比如长度为0的字符串,列表,元组,字典等)都会被当作False。因此上面的代码会误将一些其他输入也会当作没有输入。比如:

def func(a, b=None):
    if not b:
        b = []
    return b

print(func(1))                      # []
print(func(1, []))                  # []
print(func(1, 0))                   # []
print(func(1, ""))                  # []

从结果中就可以看到,b的值并没有发生改变,这不是我们想要的结果。

2.3 测试某个可选参数

当一个函数需要测试某个可选参数是否被使用者传递进来。这个时候需要小心的是不能用某个默认值,比如None,0或者False值来测试用户提供的值(因为这些值都是合法的值,是可能被用户传递进来的)。

你可以创建一个独一 无二的私有对象实例,就像上面的_no_value变量一样。在函数里面,你可以通过检查被传递参数值跟这个实例是否一样来判断。

这里的思路就是用户不可能去传递这个_no_value实例作为输出。因此这里通过检查这个值就能确定某个参数是否被传递进来了。

 

posted @ 2018-11-25 15:40  _杨魏  阅读(3361)  评论(0编辑  收藏  举报