python定义函数时默认参数注意事项

如果在调用一个函数时,没有传递默认参数,则函数内的默认参数是对函数的默认参数属性__defaults__的引用,

def func(arg1=[]):
    arg1.append(2)

调用func时如果没有传参,上面的arg1就是func.__defaults__[0]的引用

没传递默认参数,会发生以下情况

由于func.__defaults__[0]是可变类型,导致每一次调用func,arg1都会对func.__defaults__[0]进行操作(func.__defaults__[0].append(2),

这样在有些情况下会导致逻辑出错的,例如

def func(arg1=[]):
    if(arg1==[]):
        print 'arg1 is empty'
        arg1.append(1)
    else:
        print 'arg1 is not empty'
        print arg1


func() # arg1 is empty
func() #arg1 is not empty  [1]
 

第二次调用func的时候,并没有传递参数arg1,但是第一次调用时,函数内部已经修改了__defaults__

这是为啥呢?为何第二次调用不重置arg1为[]?

因为

Python的默认参数只会在函数定义时被确定,而不是每次调用时重新确定,所以,一旦在函数中修改了默认参数,则再随后的调用中都会生效

由于有这个特性,在定义函数时,如果默认参数使用可变的对象类型,如上例子,则很可能导致逻辑出错,

所以,如不是特别需要,则不允许在函数内部对默认参数引用的func.__defaults__属性进行修改,如何能让一个对象不被修改?那就是在操作arg1前取消它对__defaults__的引用

以上例子改动一下

def func(arg1=[]):
    if(arg1==[]):
        print 'arg1 is empty'
        arg1=[]
        arg1.append(1)
    else:
        print 'arg1 is not empty'
        print arg1

上例中,在用户没有传递默认参数arg1时,函数内部会给arg1变量重新赋值,让arg1去引用我们想用的对象[],这样arg1就不会修改func.__defaults__了

如果是默认参数是有值的情况,可以这样操作

def func(arg1=[1,2,3]):
    if(arg1==[1,2,3]):

        print 'is [1,2,3]'
        arg1=[1,2,3] #重点是这里,取消arg1对__defaults__属性的引用,防止arg1修改__defaults__
        arg1.append(1)

    else:
        print 'not [1,2,3]'
        print arg1

以上太啰嗦,下例模仿了python官方例子,使用不可变类型(例如None)作为默认参数,逻辑简洁,推荐使用

def append_to(element, to=None): #默认参数to定义时为None,但函数逻辑中进行进一步重新赋值为想使用的默认值
if to is None: to = [1,2,3] to.append(element) return to

 

总结:

防止默认参数修改函数的__defaults__,需要:

1.定义默认参数时,最好使用不可变类型.

2.如果默认参数一定要使用可变类型,那就在函数内部对默认参数重新赋值为可变类型的具体值.

posted @ 2017-10-22 16:03  toDoYourBest  阅读(7147)  评论(0编辑  收藏  举报