鱼C论坛

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

[已解决]无限递归死循环

[复制链接]
发表于 2023-8-29 21:23:21 | 显示全部楼层 |阅读模式

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

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

x
class D:
    def __setattr__(self,name,value):
        self.name = value
d = D()
d.name = "额"
为什么这段代码是无限递归啊?帮忙解释一下。谢谢。
最佳答案
2023-8-29 21:24:49
这段代码会导致无限递归的原因是在类D的 __setattr__ 方法中,当给 self.name 赋值时,会再次调用 __setattr__ 方法,从而导致无限递归。

具体来说,在 d.name = "额" 这行代码中,Python会调用 d 对象的 __setattr__ 方法来设置属性 name 的值。而在 __setattr__ 方法中,通过 self.name = value 语句将属性 name 的值设置为 value 。然而,这个赋值操作实际上会再次调用 __setattr__ 方法,因为在赋值操作中,Python会尝试设置 self.name 属性的值,而 self.name 又是 self 对象的属性,所以会再次调用 __setattr__ 方法,形成了递归调用。

为了避免无限递归,可以在 __setattr__ 方法中使用 super().__setattr__(name, value) 来调用父类的 __setattr__ 方法,而不是直接使用 self.name = value 。这样可以避免直接调用 __setattr__ 方法,从而解决无限递归的问题。

修正后的代码如下:
class D:
    def __setattr__(self, name, value):
        super().__setattr__(name, value)

d = D()
d.name = "额"

这样修改后,赋值操作不会再次调用 __setattr__ 方法,从而避免了无限递归。

本帖被以下淘专辑推荐:

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

使用道具 举报

发表于 2023-8-29 21:24:49 | 显示全部楼层    本楼为最佳答案   
这段代码会导致无限递归的原因是在类D的 __setattr__ 方法中,当给 self.name 赋值时,会再次调用 __setattr__ 方法,从而导致无限递归。

具体来说,在 d.name = "额" 这行代码中,Python会调用 d 对象的 __setattr__ 方法来设置属性 name 的值。而在 __setattr__ 方法中,通过 self.name = value 语句将属性 name 的值设置为 value 。然而,这个赋值操作实际上会再次调用 __setattr__ 方法,因为在赋值操作中,Python会尝试设置 self.name 属性的值,而 self.name 又是 self 对象的属性,所以会再次调用 __setattr__ 方法,形成了递归调用。

为了避免无限递归,可以在 __setattr__ 方法中使用 super().__setattr__(name, value) 来调用父类的 __setattr__ 方法,而不是直接使用 self.name = value 。这样可以避免直接调用 __setattr__ 方法,从而解决无限递归的问题。

修正后的代码如下:
class D:
    def __setattr__(self, name, value):
        super().__setattr__(name, value)

d = D()
d.name = "额"

这样修改后,赋值操作不会再次调用 __setattr__ 方法,从而避免了无限递归。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-8-29 21:43:11 | 显示全部楼层
因为 self.name = value 同样会触发自身的 __setattr__() 魔法方法。

点评

哇谢谢小甲鱼哥。刚才在看你的视频呢。  发表于 2023-8-29 22:08
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

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

本版积分规则

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

GMT+8, 2024-9-21 15:26

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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