鱼C论坛

 找回密码
 立即注册
查看: 1535|回复: 3

[已解决]属性访问魔法方法中死循环的理解

[复制链接]
发表于 2023-6-19 17:16:42 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
>>> class Rectangle:
...     def __init__(self,width=0,height=0):
...         self.width=width
...         self.height=height
...     def __setattr__(self,name,value):
...         if name=='square':
...             self.width=value
...             self.height=value
...         else:
...             self.name=value
...     def getarea(self):
...         return self.width*self.height

else下一句self.name=value,这里会在实例化对象时触发死循环,我大概能理解
但是if name=='square':下面的self.width=value,self.height=value,应该也会触发死循环吧,当square赋值时,触发__setattr__()魔法方法,执行self.width=value,self.height=value,这里执行会再次触发__setattr__()魔法方法。
可实际上没报错,这是为什么
最佳答案
2023-6-19 17:31:11
你的理解是对的,这段代码中存在一个问题,它的确会导致无限递归,这是因为  __setattr__  方法会在尝试对属性进行赋值操作时被调用,如果在  __setattr__  方法内部再次对属性进行赋值,就会再次调用  __setattr__ ,从而形成无限递归。

然而,这个代码块的行为可能取决于你使用的 Python 的具体版本或实现。在某些版本或实现中,这可能不会引发错误,因为 Python 有一些内部机制可能会防止这种无限递归。

然而,最佳的做法是避免这种情况的发生。如果你希望在设置属性值时执行某些操作,你可以使用 property 装饰器来实现,而不是覆盖  __setattr__ 。例如:
class Rectangle:
    def __init__(self, width=0, height=0):
        self._width = width
        self._height = height

    @property
    def width(self):
        return self._width

    @width.setter
    def width(self, value):
        self._width = value

    @property
    def height(self):
        return self._height

    @height.setter
    def height(self, value):
        self._height = value

    @property
    def square(self):
        return self._width * self._height

    @square.setter
    def square(self, value):
        self._width = value
        self._height = value

    def getarea(self):
        return self._width * self._height

在这个版本中,我使用了前导下划线的属性  _width  和  _height  来存储宽度和高度的值。当你尝试获取  width 、 height  或  square  的值时,会返回这些内部属性的值;当你尝试设置这些属性的值时,会更新内部属性的值,而不会引发无限递归。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2023-6-19 17:31:11 | 显示全部楼层    本楼为最佳答案   
你的理解是对的,这段代码中存在一个问题,它的确会导致无限递归,这是因为  __setattr__  方法会在尝试对属性进行赋值操作时被调用,如果在  __setattr__  方法内部再次对属性进行赋值,就会再次调用  __setattr__ ,从而形成无限递归。

然而,这个代码块的行为可能取决于你使用的 Python 的具体版本或实现。在某些版本或实现中,这可能不会引发错误,因为 Python 有一些内部机制可能会防止这种无限递归。

然而,最佳的做法是避免这种情况的发生。如果你希望在设置属性值时执行某些操作,你可以使用 property 装饰器来实现,而不是覆盖  __setattr__ 。例如:
class Rectangle:
    def __init__(self, width=0, height=0):
        self._width = width
        self._height = height

    @property
    def width(self):
        return self._width

    @width.setter
    def width(self, value):
        self._width = value

    @property
    def height(self):
        return self._height

    @height.setter
    def height(self, value):
        self._height = value

    @property
    def square(self):
        return self._width * self._height

    @square.setter
    def square(self, value):
        self._width = value
        self._height = value

    def getarea(self):
        return self._width * self._height

在这个版本中,我使用了前导下划线的属性  _width  和  _height  来存储宽度和高度的值。当你尝试获取  width 、 height  或  square  的值时,会返回这些内部属性的值;当你尝试设置这些属性的值时,会更新内部属性的值,而不会引发无限递归。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-6-19 17:43:08 | 显示全部楼层
isdkz 发表于 2023-6-19 17:31
你的理解是对的,这段代码中存在一个问题,它的确会导致无限递归,这是因为  __setattr__  方法会在尝试对 ...

回答的好详细呀,还给出了解决办法
非常感谢
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-6-19 22:38:06 | 显示全部楼层
isdkz 发表于 2023-6-19 17:31
你的理解是对的,这段代码中存在一个问题,它的确会导致无限递归,这是因为  __setattr__  方法会在尝试对 ...

wc
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-12-23 17:37

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表