您需要了解默认值的工作原理才能有效地使用它们。
函数是对象。因此,它们具有属性。所以,如果我创建这个函数:
>>> def f(x, y=[]):
y.append(x)
return y
我创建了一个对象。以下是它的属性:
>>> dir(f)
['__call__', '__class__', '__closure__', '__code__', '__defaults__', '__delattr__',
'__dict__', '__doc__', '__format__', '__get__', '__getattribute__', '__globals__',
'__hash__', '__init__', '__module__', '__name__', '__new__', '__reduce__',
'__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__',
'func_closure', 'func_code', 'func_defaults', 'func_dict', 'func_doc', 'func_globals',
'func_name']
其中之一是func_defaults
。听起来很有希望,里面有什么?
>>> f.func_defaults
([],)
这是一个包含函数默认值的元组。如果默认值是一个对象,则元组包含该对象的一个实例。
如果您正在考虑将一个项目添加到列表中,这会导致一些相当违反直觉的行为f
,如果没有提供列表,则返回一个仅包含该项目的列表:
>>> f(1)
[1]
>>> f(2)
[1, 2]
但是,如果您知道默认值是存储在函数属性之一中的对象实例,那么它就不那么违反直觉了:
>>> x = f(3)
>>> y = f(4)
>>> x == y
True
>>> x
[1, 2, 3, 4]
>>> x.append(5)
>>> f(6)
[1, 2, 3, 4, 5, 6]
知道了这一点,很明显,如果您希望函数参数的默认值是一个新列表(或任何新对象),您不能简单地将对象的实例存储在func_defaults
. 每次调用该函数时,您都必须创建一个新函数:
>>>def g(x, y=None):
if y==None:
y = []
y.append(x)
return y