Python 可变类型作为函数默认参数时的副作用

简介: Python 可变类型作为函数默认参数时的副作用

在 Python 中定义函数时,可以为其指定 默认参数,这样就不必在每次调用函数时都传递参数进去,并且可以简化我们的代码。

在定义函数时,如果使用了 可变类型 来作为函数的 默认参数,往往会产生一些副作用。来看下面一段代码。

def foo(li=[]):
    li.append(1)
    print(li)
foo()
foo()
foo()

你可能想得到如下的结果:

[1]
[1]
[1]

但实际上,结果却是:

[1]
[1, 1]
[1, 1, 1]

根据结果来看,似乎每次的函数调用,li 列表都记录了上一次的调用结果,而并没有使用默认参数 []。但是当我们在每次调用函数时,都传递一个空列表 [] 进去,就不会出现副作用,能够正确得到我们想要的结果。

def foo(li=[]):
    li.append(1)
    print(li)
foo([])
foo([])
foo([])
[1]
[1]
[1]

为什么会产生这样的结果呢?其实这个问题的原因是 Python 中函数的特性所决定的。由于函数也是 对象,而 对象 可以有自己的属性,所以函数也有自己的属性。

foo 函数被创建时,它的 默认参数 就已经被保存在它的 __defaults__ 属性中了。而函数只会被创建一次,以后每次执行 foo() 的时候,都只会调用函数,并不会去重新创建一个函数。所以函数的 默认参数 也只会计算一次,无论之后被调用多少次,默认参数 始终都是同一个 对象

我们用 id() 函数打印出默认参数的内存地址就能够看出来。

def foo(li=[]):
    li.append(1)
    print(li, id(li))
foo()
foo()
foo()
[1] 48904632
[1, 1] 48904632
[1, 1, 1] 48904632

可以看出,三次调用函数后,其内部打印的 li 地址是相同的,所以它们其实是同一个 对象

知道了问题的原因,那么如果解决它呢?我们可以用 None 来作为函数的 默认参数,在调用 foo 函数时,在函数体内部可以通过判断 li 是否为 None 来决定是否需要使用 默认值。这样,当调用 foo() 函数并且没有传递参数时,再给 li 赋一个 默认值 即可。

def foo(li=None):
    if li is None:
        li = []
    li.append(1)
    print(li)
foo()
foo()
foo()


[1]
[1]
[1]

这样既不用担心 可变类型 作为函数 默认参数 时的副作用,也不用在每次调用 foo 函数时都传递一个参数进去,能够很好的解决这个问题。

所以,当我们给函数定义 默认参数 时,应该尽量使用 不可变类型 以免产生意想不到的副作用。当然,除非你明确知道你需要用 可变类型 的特性来达到某些目的。


相关文章
|
7月前
|
存储 JavaScript Java
(Python基础)新时代语言!一起学习Python吧!(四):dict字典和set类型;切片类型、列表生成式;map和reduce迭代器;filter过滤函数、sorted排序函数;lambda函数
dict字典 Python内置了字典:dict的支持,dict全称dictionary,在其他语言中也称为map,使用键-值(key-value)存储,具有极快的查找速度。 我们可以通过声明JS对象一样的方式声明dict
449 2
|
7月前
|
算法 Java Docker
(Python基础)新时代语言!一起学习Python吧!(三):IF条件判断和match匹配;Python中的循环:for...in、while循环;循环操作关键字;Python函数使用方法
IF 条件判断 使用if语句,对条件进行判断 true则执行代码块缩进语句 false则不执行代码块缩进语句,如果有else 或 elif 则进入相应的规则中执行
1300 1
|
7月前
|
Java 数据处理 索引
(numpy)Python做数据处理必备框架!(二):ndarray切片的使用与运算;常见的ndarray函数:平方根、正余弦、自然对数、指数、幂等运算;统计函数:方差、均值、极差;比较函数...
ndarray切片 索引从0开始 索引/切片类型 描述/用法 基本索引 通过整数索引直接访问元素。 行/列切片 使用冒号:切片语法选择行或列的子集 连续切片 从起始索引到结束索引按步长切片 使用slice函数 通过slice(start,stop,strp)定义切片规则 布尔索引 通过布尔条件筛选满足条件的元素。支持逻辑运算符 &、|。
382 0
|
8月前
|
设计模式 缓存 监控
Python装饰器:优雅增强函数功能
Python装饰器:优雅增强函数功能
364 101
|
8月前
|
缓存 测试技术 Python
Python装饰器:优雅地增强函数功能
Python装饰器:优雅地增强函数功能
302 99
|
8月前
|
存储 缓存 测试技术
Python装饰器:优雅地增强函数功能
Python装饰器:优雅地增强函数功能
498 98
|
8月前
|
JSON 缓存 开发者
淘宝商品详情接口(item_get)企业级全解析:参数配置、签名机制与 Python 代码实战
本文详解淘宝开放平台taobao.item_get接口对接全流程,涵盖参数配置、MD5签名生成、Python企业级代码实现及高频问题排查,提供可落地的实战方案,助你高效稳定获取商品数据。
|
算法 Python 容器
Python编程 - 不调用相关choose库函数,“众数“挑选器、随机挑选器 的源码编程实现
Python编程 - 不调用相关choose库函数,“众数“挑选器、随机挑选器 的源码编程实现
348 0
|
算法 Python
Python编程的函数—内置函数
Python编程的函数—内置函数
330 1
|
算法 Python
Python编程实验四:函数的使用
Python编程实验四:函数的使用

推荐镜像

更多