°

python 默认参数设置一次问题

2019年课程规划及价格说明


PHP实战视频教程大全

一个函数参数的默认值,仅仅在该函数定义的时候,被赋值一次

首先来看问题:

>>> def add_end(L=[]):
...     L.append('END')
...     return L
...
>>>
>>> t=add_end()
>>>
>>> t
['END']
>>> t=add_end()
>>> t
['END', 'END']
>>> t=add_end([1])
>>> t
[1, 'END']
>>> t=add_end()
>>> t
['END', 'END', 'END']
>>> t=add_end([1])
>>> t
[1, 'END']
>>> t=add_end()
>>> t
['END', 'END', 'END', 'END']

Python函数在定义的时候(只在那个时刻赋值,此后调用不再赋值),默认参数L的值就被计算出来了,即[],因为默认参数L也是一个变量,它指向对象[],每次调用该函数,如果改变了L的内容,则下次调用时,默认参数的内容就变了,不再是函数定义时的[]了

Python函数参数对于可变对象,函数内对参数的改变会影响到原始对象;对于不可变对象,函数内对参数的改变不会影响到原始参数。原因在于:

1、可变对象,参数改变的是可变对象,其内容可以被修改。

2、不可变对象,改变的是函数内变量的指向对象。

更明显的例子:可以看出初始化时机

import datetime as dt

def log_time(message, time=dt.datetime.now()):
print("{0}: {1}".format(time.isoformat(), message))

每次我们调用log_time()时都期望它能够正确提供当前时间。悲剧的是并没有成功:默认参数在定义时求值(比如说当你首次导入模块时),调用的结果如下

>>> log_time("message 1")
2015-02-10T21:20:32.998647: message 1
>>> log_time("message 2")
2015-02-10T21:20:32.998647: message 2
>>> log_time("message 3")
2015-02-10T21:20:32.998647: message 3

对于不可变对象,他的内内容发生改变时,变量的对象引用是不会变化的,所以上面的L一直是同一个地址的L,所以会不断的家END 而当改成如下形式:

def add_end(L=None):
    if L is None:
        L = []
    L.append('END')
    return L

现在,无论调用多少次,都不会有问题。

None 是不可变对象,每次一赋值为[]时,他指向的引用就会发生改变,输出的结果也是改变后的引用内容,但再一次调用L还是之前的引用(None)
所以,定义默认参数要牢记一点:默认参数必须指向不变对象!

打赏
  喜欢