Python的类成员变量默认初始值的坑及解决

Anna ·
更新时间:2024-11-10
· 1278 次阅读

目录

类成员变量默认初始值的坑

问题发现

示例代码

结果

原因

解决方法

Python默认值参数

简单粗暴上代码

可以用下面这种写法

类成员变量默认初始值的坑 问题发现

一个循环内,缺省值初始化同名变量,其中的list成员不是空,会延续之前同名变量的值。

示例代码 # Define class class Variant():     # use     def __init__(self, price = 500, description = 'default description', values = ['', '', '']):         self.price = price         self.description = description         self.values = values     def __str__(self):         return 'price: {}, description: {}, values: {}'.format(self.price, self.description, self.values) variant_list = [] # Create instance with same name iteratively for i in range(3):     current_variant = Variant()     if i == 1:         current_variant.values[2] = 'hello'     current_variant.price = i     current_variant.description = 'description of variant: {}'.format(i)     variant_list.append(current_variant) # Test results for variant in variant_list:     print(str(variant)) 结果

所有实例的values列表值相同

原因

可选参数默认值的设置在Python中只会被执行一次,也就是定义该函数的时候”如此使用缺省值初始化,list成员指向的是同一个list(地址),如果只是修改其中一个元素(而不是赋值新的list开辟新内存),那么所有instance的list成员都会被修改。

解决方法

直接在构造方法中置为空(self.values = ['', '', '']),之后各个修改值

Python默认值参数 简单粗暴上代码 def fun(a, b=[]):     b += [a]     print(b) fun(1) fun(2,[]) fun(3)

是不是看上去很简单,其实暗藏玄机,请大家看一下输出结果,是不是有点让你疑惑^^~

[1]
[2]
[1, 3]

此时你是否也和我有一样的疑惑,为什么 fun(3) 的输出结果是 [1, 3]?

哈哈,不卖关子了,这里是因为,因为函数被定义好后,只会生成一次,所以在函数生成的时候定义的变量 b 的默认值也只会被初始化一次。

因此,当执行fun(1)函数时,没有给 b 传参,所以使用的是 b 的默认值,此时 b 的默认值为[1]。

执行fun(2,[])时,给 b 传了一个[]值(恰好和默认值相同,其实是不同的数据),因此便使用的是传入数据,执行结果便是[2]。

然后在执行fun(3),此刻又没有给 b 传参,所以依旧使用的是 b 的默认值, 而 b 的默认值只会随着函数的生成被生成一次 ( fun(1) 生成过了 ),所以现在的默认值是fun(1)的执行结果[1],因此当fun(3)再次调用时,输出结果便会是[1, 3]。

如果不行出现当前这种情况,而是在函数每次被调用的时候都初始化一次变量

可以用下面这种写法 def function(a, b=None):     b = b if b else []  # 明确每次重新定义b     b += [a]     print(b) function(1) function(2, []) function(3)

输出结果:

[1]
[2]
[3]

以上为个人经验,希望能给大家一个参考,也希望大家多多支持软件开发网。



python的 默认 成员变量 Python 变量

需要 登录 后方可回复, 如果你还没有账号请 注册新账号