【Python】解决类中特性(property)覆盖同名属性(attribute)报错问题
创始人
2024-03-16 06:08:28

一、问题描述

在设计一类加法的数据结构中,定义 Foo 对象可以拥有自己的取值 value 以及一系列可以通过累加得到该 Foo 对象的取值 value 的加法因子 factors

为简化对 Foo 对象的维护,希望每次获取 Foo 的取值 value 时,可以直接通过其加法因子列表元素的累加来获取其 value,而不需要每次手动对其 value 进行更新,因此考虑通过装饰器 @property 来实现该功能,如以下代码所示。

class Foo:def __init__(self, value, factors=None):self.value = valueself.factors = [self] if factors is None else factors@propertydef value(self):return sum([factor.value for factor in self.factors]) if len(self.factors) > 1 else self.valuefoo1 = Foo(value=3)
foo2 = Foo(value=5)
foo = Foo(value=8, factors=[foo1, foo2])print(f"The value of foo1 is: {foo1.value}.")
print(f"The value of foo2 is: {foo2.value}.")
print(f"The value of foo  is: {foo.value}.")

然而,运行该段代码后,程序报错 AttributeError,详细报错信息如下:

Traceback (most recent call last):File "attribute_property.py", line 11, in foo1 = Foo(value=3)File "attribute_property.py", line 3, in __init__self.value = value
AttributeError: can't set attribute

二、问题分析

出现该报错的原因是因为 property 装饰器作为特性重写了 value 属性,因此当创建新的 Foo 对象时,line 3 实际上将找到 @property 重写的 value 方法,而由于 @property 实际上只实现了 @getter 功能,因此可认为它只实现了可读属性的功能,而并没有实现 @setter 的功能。简单来说,这个时候是不能为它赋值的。(关于 @getter@setter 等相关装饰器内容请参见相关博客。)

三、解决方案

从上例中可以看到,此时我们是无法使用特性(property)直接覆盖同名属性(attribute)的。

一个简单的解决方案是将原来的 value 属性设置为保护变量,即在其变量名前添加下划线成为 _value。对应地,将 @property 实现的类方法中所返回的 value 属性更改为 _value。既然此时这二者都不同名了,自然也不会出现设置属性报错的问题了。

修改后的代码及运行结果如下所示:

class Foo:def __init__(self, value, factors=None):self._value = valueself.factors = [self] if factors is None else factors@propertydef value(self):return sum([factor.value for factor in self.factors]) if len(self.factors) > 1 else self._valuefoo1 = Foo(value=3)
foo2 = Foo(value=5)
foo = Foo(value=8, factors=[foo1, foo2])print(f"The value of foo1 is: {foo1.value}.")
print(f"The value of foo2 is: {foo2.value}.")
print(f"The value of foo  is: {foo.value}.")
---------------------------------------------
The value of foo1 is: 3.
The value of foo2 is: 5.
The value of foo  is: 8.

相关内容

热门资讯

拟剥离地产开发业务,60068... 一家国资背景的房企,拟剥离地产开发业务。3月15日晚间,京投发展(600683)发布公告称,拟将持有...
一些信息 0315 (来源:渾水調研)抱歉回来晚了1、美伊。消息错综复杂,但互联网上各种意见领袖的神贴,进一步强化中国新...
韩国综合指数和台湾加权指数为何... 人工智能浪潮,已经被举世公认为人类历史上继机械化、自动化、信息化后的第四次工业革命。在资本市场投资机...
高市早苗病倒了,多方压力打垮了... (来源:时报新征途)  日本政坛这两天突然冒出一个耐人寻味的消息:高市早苗在国会会议结束后起身困难,...
合肥持续开展低速电动四轮车专项... (来源:市场星报) 星报讯 低速电动四轮车,俗称“老头乐”。此类车辆无安全装置,车身脆弱,不符合国...